What's the difference between abstraction and generalization?

后端 未结 8 1159
滥情空心
滥情空心 2021-01-30 13:26

I understand that abstraction is about taking something more concrete and making it more abstract. That something may be either a data structure or a procedure. For example:

相关标签:
8条回答
  • 2021-01-30 13:45

    Let me explain in the simplest manner possible.

    "All pretty girls are female." is an abstraction.

    "All pretty girls put on make-up." is a generalization.

    0 讨论(0)
  • 2021-01-30 13:46

    Not addressing credible / official source: an example in Scala

    Having "Abstraction"

      trait AbstractContainer[E] { val value: E }
    
      object StringContainer extends AbstractContainer[String] {
        val value: String = "Unflexible"
      }
    
      class IntContainer(val value: Int = 6) extends AbstractContainer[Int]
    
      val stringContainer = new AbstractContainer[String] {
        val value = "Any string"
      }
    

    and "Generalization"

      def specialized(c: StringContainer.type) =
        println("It's a StringContainer: " + c.value)
    
      def slightlyGeneralized(s: AbstractContainer[String]) =
        println("It's a String container: " + s.value)
    
      import scala.reflect.{ classTag, ClassTag }
      def generalized[E: ClassTag](a: AbstractContainer[E]) =
        println(s"It's a ${classTag[E].toString()} container: ${a.value}")
    
      import scala.language.reflectiveCalls
      def evenMoreGeneral(d: { def detail: Any }) =
        println("It's something detailed: " + d.detail)
    

    executing

      specialized(StringContainer)
      slightlyGeneralized(stringContainer)
      generalized(new IntContainer(12))
      evenMoreGeneral(new { val detail = 3.141 })
    

    leads to

    It's a StringContainer: Unflexible
    It's a String container: Any string
    It's a Int container: 12
    It's something detailed: 3.141
    
    0 讨论(0)
  • 2021-01-30 13:47

    A very interesting question indeed. I found this article on the topic, which concisely states that:

    While abstraction reduces complexity by hiding irrelevant detail, generalization reduces complexity by replacing multiple entities which perform similar functions with a single construct.

    Lets take the old example of a system that manages books for a library. A book has tons of properties (number of pages, weight, font size(s), cover,...) but for the purpose of our library we may only need

    Book(title, ISBN, borrowed)
    

    We just abstracted from the real books in our library, and only took the properties that interested us in the context of our application.


    Generalization on the other hand does not try to remove detail but to make functionality applicable to a wider (more generic) range of items. Generic containers are a very good example for that mindset: You wouldn't want to write an implementation of StringList, IntList, and so on, which is why you'd rather write a generic List which applies to all types (like List[T] in Scala). Note that you haven't abstracted the list, because you didn't remove any details or operations, you just made them generically applicable to all your types.

    Round 2

    @dtldarek's answer is really a very good illustration! Based on it, here's some code that might provide further clarification.

    Remeber the Book I mentioned? Of course there are other things in a library that one can borrow (I'll call the set of all those objects Borrowable even though that probably isn't even a word :D):

    All of these items will have an abstract representation in our database and business logic, probably similar to that of our Book. Additionally, we might define a trait that is common to all Borrowables:

    trait Borrowable {
        def itemId:Long
    }
    

    We could then write generalized logic that applies to all Borrowables (at that point we don't care if its a book or a magazine):

    object Library {
        def lend(b:Borrowable, c:Customer):Receipt = ...
        [...]
    }
    

    To summarize: We stored an abstract representation of all the books, magazines and DVDs in our database, because an exact representation is neither feasible nor necessary. We then went ahead and said

    It doesn't matter whether a book, a magazine or a DVD is borrowed by a customer. It's always the same process.

    Thus we generalized the operation of borrowing an item, by defining all things that one can borrow as Borrowables.

    0 讨论(0)
  • 2021-01-30 14:03

    Object:

    portal cake photo

    Abstraction:

    cake symbol

    Generalization:

    many desserts

    Example in Haskell:

    The implementation of the selection sort by using priority queue with three different interfaces:

    • an open interface with the queue being implemented as a sorted list,
    • an abstracted interface (so the details are hidden behind the layer of abstraction),
    • a generalized interface (the details are still visible, but the implementation is more flexible).
    {-# LANGUAGE RankNTypes #-}
    
    module Main where
    
    import qualified Data.List as List
    import qualified Data.Set as Set
    
    {- TYPES: -}
    
    -- PQ new push pop
    -- by intention there is no build-in way to tell if the queue is empty
    data PriorityQueue q t = PQ (q t) (t -> q t -> q t) (q t -> (t, q t))
    -- there is a concrete way for a particular queue, e.g. List.null
    type ListPriorityQueue t = PriorityQueue [] t
    -- but there is no method in the abstract setting
    newtype AbstractPriorityQueue q = APQ (forall t. Ord t => PriorityQueue q t)
    
    
    {- SOLUTIONS: -}
    
    -- the basic version
    list_selection_sort :: ListPriorityQueue t -> [t] -> [t]
    list_selection_sort (PQ new push pop) list = List.unfoldr mypop (List.foldr push new list)
      where
        mypop [] = Nothing -- this is possible because we know that the queue is represented by a list
        mypop ls = Just (pop ls)
    
    
    -- here we abstract the queue, so we need to keep the queue size ourselves
    abstract_selection_sort :: Ord t => AbstractPriorityQueue q -> [t] -> [t]
    abstract_selection_sort (APQ (PQ new push pop)) list = List.unfoldr mypop (List.foldr mypush (0,new) list)
      where
        mypush t (n, q) = (n+1, push t q)
        mypop (0, q) = Nothing
        mypop (n, q) = let (t, q') = pop q in Just (t, (n-1, q'))
    
    
    -- here we generalize the first solution to all the queues that allow checking if the queue is empty
    class EmptyCheckable q where
      is_empty :: q -> Bool
    
    generalized_selection_sort :: EmptyCheckable (q t) => PriorityQueue q t -> [t] -> [t]
    generalized_selection_sort (PQ new push pop) list = List.unfoldr mypop (List.foldr push new list)
      where
        mypop q | is_empty q = Nothing
        mypop q | otherwise  = Just (pop q)
    
    
    {- EXAMPLES: -}
    
    -- priority queue based on lists
    priority_queue_1 :: Ord t => ListPriorityQueue t
    priority_queue_1 = PQ [] List.insert (\ls -> (head ls, tail ls))
    instance EmptyCheckable [t] where
      is_empty = List.null
    
    -- priority queue based on sets
    priority_queue_2 :: Ord t => PriorityQueue Set.Set t
    priority_queue_2 = PQ Set.empty Set.insert Set.deleteFindMin
    instance EmptyCheckable (Set.Set t) where
      is_empty = Set.null
    
    -- an arbitrary type and a queue specially designed for it
    data ABC = A | B | C deriving (Eq, Ord, Show)
    
    -- priority queue based on counting
    data PQ3 t = PQ3 Integer Integer Integer
    priority_queue_3 :: PriorityQueue PQ3 ABC
    priority_queue_3 = PQ new push pop
      where
        new = (PQ3 0 0 0)
        push A (PQ3 a b c) = (PQ3 (a+1) b c)
        push B (PQ3 a b c) = (PQ3 a (b+1) c)
        push C (PQ3 a b c) = (PQ3 a b (c+1))
        pop (PQ3 0 0 0) = undefined
        pop (PQ3 0 0 c) = (C, (PQ3 0 0 (c-1)))
        pop (PQ3 0 b c) = (B, (PQ3 0 (b-1) c))
        pop (PQ3 a b c) = (A, (PQ3 (a-1) b c))
    
    instance EmptyCheckable (PQ3 t) where
      is_empty (PQ3 0 0 0) = True
      is_empty _ = False
    
    
    {- MAIN: -}
    
    main :: IO ()
    main = do
      print $ list_selection_sort priority_queue_1 [2, 3, 1]
      -- print $ list_selection_sort priority_queue_2 [2, 3, 1] -- fail
      -- print $ list_selection_sort priority_queue_3 [B, C, A] -- fail
      print $ abstract_selection_sort (APQ priority_queue_1) [B, C, A] -- APQ hides the queue 
      print $ abstract_selection_sort (APQ priority_queue_2) [B, C, A] -- behind the layer of abstraction
      -- print $ abstract_selection_sort (APQ priority_queue_3) [B, C, A] -- fail
      print $ generalized_selection_sort priority_queue_1 [2, 3, 1]
      print $ generalized_selection_sort priority_queue_2 [B, C, A]
      print $ generalized_selection_sort priority_queue_3 [B, C, A]-- power of generalization
    
      -- fail
      -- print $ let f q = (list_selection_sort q [2,3,1], list_selection_sort q [B,C,A])
      --         in f priority_queue_1
    
      -- power of abstraction (rank-n-types actually, but never mind)
      print $ let f q = (abstract_selection_sort q [2,3,1], abstract_selection_sort q [B,C,A]) 
              in f (APQ priority_queue_1)
    
      -- fail
      -- print $ let f q = (generalized_selection_sort q [2,3,1], generalized_selection_sort q [B,C,A])
      --         in f priority_queue_1
    

    The code is also available via pastebin.

    Worth noticing are the existential types. As @lukstafi already pointed out, abstraction is similar to existential quantifier and generalization is similar to universal quantifier. Observe that there is a non-trivial connection between the fact that ∀x.P(x) implies ∃x.P(x) (in a non-empty universe), and that there rarely is a generalization without abstraction (even c++-like overloaded functions form a kind of abstraction in some sense).

    Credits: Portal cake by Solo. Dessert table by djttwo. The symbol is the cake icon from material.io.

    0 讨论(0)
  • 2021-01-30 14:03

    I'm going to use some examples to describe generalisation and abstraction, and I'm going to refer to this article.

    To my knowledge, there is no official source for the definition of abstraction and generalisation in the programming domain (Wikipedia is probably the closest you'll get to an official definition in my opinion), so I've instead used an article which I deem credible.

    Generalization

    The article states that:

    "The concept of generalization in OOP means that an object encapsulates common state and behavior for a category of objects."

    So for example, if you apply generalisation to shapes, then the common properties for all types of shape are area and perimeter.

    Hence a generalised shape (e.g. Shape) and specialisations of it (e.g. a Circle), can be represented in classes as follows (note that this image has been taken from the aforementioned article)

    enter image description here

    Similarly, if you were working in the domain of jet aircraft, you could have a Jet as a generalisation, which would have a wingspan property. A specialisation of a Jet could be a FighterJet, which would inherit the wingspan property and would have its own property unique to fighter jets e.g. NumberOfMissiles.

    Abstraction

    The article defines abstraction as:

    "the process of identifying common patterns that have systematic variations; an abstraction represents the common pattern and provides a means for specifying which variation to use" (Richard Gabriel)"

    In the domain of programming:

    An abstract class is a parent class that allows inheritance but can never be instantiated.

    Hence in the example given in the Generalization section above, a Shape is abstract as:

    In the real world, you never calculate the area or perimeter of a generic shape, you must know what kind of geometric shape you have because each shape (eg. square, circle, rectangle, etc.) has its own area and perimeter formulas.

    However, as well as being abstract a shape is also a generalisation (because it "encapsulates common state and behavior for a category of objects" where in this case the objects are shapes).

    Going back to the example I gave about Jets and FighterJets, a Jet is not abstract as a concrete instance of a Jet is feasible, as one can exist in the real world, unlike a shape i.e. in the real world you cant hold a shape you hold an instance of a shape e.g. a cube. So in the aircraft example, a Jet is not abstract, it is a generalisation as it is possible to have a "concrete" instance of a jet.

    0 讨论(0)
  • 2021-01-30 14:05

    Abstraction is usually about reducing complexity by eliminating unnecessary details. For example, an abstract class in OOP is a parent class that contains common features of its children but does not specify the exact functionality.

    Generalization does not necessarily require to avoid details but rather to have some mechanism to allow for applying the same function to different argument. For instance, polymorphic types in functional programming languages allow you not to bother about the arguments, rather focus on the operation of the function. Similarly, in java you can have generic type which is an "umbrella" to all types while the function is the same.

    0 讨论(0)
提交回复
热议问题