Composition, Aggregation, Inheritance
Properly choice between Composition, Aggregation, and Inheritance, for User Defined Types.
Preamble:
Complex type structures can be built through composition, aggregation, and inheritance.
Composition or aggregation (specialized form of association) is the process of creating more complex types from simple ones, while inheritance is the process of creating more complex type by acquiring the attributes and behaviors of existing ones.
There is a very famous design principle which says “Favor composition over inheritance“, or also "Don’t use inheritance just to get code reuse".
One other reason to favor composition with FreeBASIC (as with many other languages) is it does not support multiple inheritance.
Composition or aggregation (specialized form of association) is the process of creating more complex types from simple ones, while inheritance is the process of creating more complex type by acquiring the attributes and behaviors of existing ones.
There is a very famous design principle which says “Favor composition over inheritance“, or also "Don’t use inheritance just to get code reuse".
One other reason to favor composition with FreeBASIC (as with many other languages) is it does not support multiple inheritance.
Definition of composition, aggregation, inheritance
The right question to ask himself for choosing the type of relationship to use is to determine if this relationship is rather "HAS-A" or "USES-A" or "IS-A" throughout the lifetime of the application (see example below).
When user needs to use property and behavior of a type without modifying it inside its type, then association is a better option.
Whereas if needed to use and modify property and behavior of a type inside the user type, it is best to use inheritance.
Definition of each relationship
When user needs to use property and behavior of a type without modifying it inside its type, then association is a better option.
Whereas if needed to use and modify property and behavior of a type inside the user type, it is best to use inheritance.
Definition of each relationship
Composition (strong association) establishes a "HAS-A" relationship between objects:
- In composition, the component object exists solely for the use of its composite object.
- If the composite object is destroyed, the component object is destroyed as well:
- Syntax of the member field to add in the type for object's composition (or similar syntax):
Aggregation (weak association) establishes a “USES-A” relationship between objects:- If the composite object is destroyed, the component object is destroyed as well:
> because the type contains the object itself.
- In aggregation, the component object can have an existence independent of its use in the aggregate object.
- Destroying the aggregate does not destroy the component:
- Syntax of the member field to add in the type for object's aggregation (or similar syntax):
Inheritance establishes an “IS-A” relationship between instantiable types:- Destroying the aggregate does not destroy the component:
> because the type contains only a pointer to the object.
Advantages of object association (composition/aggregation) over inheritance
In most cases "HAS-A" or "USES-A" relationship is more semantically correct than "IS-A" relationship.
Aggregation is more flexible than inheritance.
Implementation of type can be changed at run-time by changing the component object, but this cannot be do with inheritance (behavior of base type cannot be changed at run-time).
A design based on object association usually will have less types.
It is possible to implement a pseudo "multiple inheritance" (in languages which do not support it) by composing multiple objects into one.
There is no conflict between procedures/properties names, which might occur with inheritance.
Aggregation is more flexible than inheritance.
Implementation of type can be changed at run-time by changing the component object, but this cannot be do with inheritance (behavior of base type cannot be changed at run-time).
A design based on object association usually will have less types.
It is possible to implement a pseudo "multiple inheritance" (in languages which do not support it) by composing multiple objects into one.
There is no conflict between procedures/properties names, which might occur with inheritance.
Downsides of object association (composition/aggregation) compared to inheritance
When using aggregation, the behavior of the system may be harder to understand just by looking at the source code, since it's more dynamic and more interaction between types happens in run-time, rather than compile time.
Association approach might require more code and time effort.
A design based on object association usually will have more objects.
Association approach might require more code and time effort.
A design based on object association usually will have more objects.
Example
Simple illustrative example that mixes composition, aggregation, and inheritance:
See also
' Between the different types "Driver", "Person", "Driver_license" and "Vehicle", the respective relationships are:
' - A driver “IS-A” person (driver is a person): => "INHERITANCE".
' - A driver “HAS-A” driver's license (driver license only existing for the driver): => "COMPOSITION".
' - A driver “USES-A” vehicle (vehicle lifetime independent of the driver life): => "AGGREGATION".
Type Person
Public:
Dim As String full_name
Declare Constructor (ByRef _full_name As String)
Protected: '' to forbid at compile time the default-construction attempt of a Person instance
Declare Constructor ()
End Type
Constructor Person (ByRef _full_name As String)
This.full_name = _full_name
End Constructor
Type Driver_license
Public:
Dim As Integer number
End Type
Type Vehicle
Public:
Dim As String registration
Declare Constructor (ByRef _registration As String)
End Type
Constructor Vehicle (ByRef _registration As String)
This.registration = _registration
End Constructor
Type Driver Extends Person '' inheritance
Public:
Dim As Driver_license dl '' composition
Dim As Vehicle Ptr pv '' aggregation
Declare Constructor (ByRef _full_name As String, ByRef _dl As Driver_license)
End Type
Constructor Driver (ByRef _full_name As String, ByRef _dl As Driver_license)
Base(_full_name)
This.dl = _dl
End Constructor
Dim As Driver d1 = Driver("User fxm", Type<Driver_license>(123456789))
Dim As Vehicle Ptr pv1 = New Vehicle("ABCDEFGHI")
d1.pv = pv1
Print "Person full name : " & d1.full_name
Print "Driver license number : " & d1.dl.number
Print "Vehicle registration : " & d1.pv->registration
Delete pv1
d1.pv = 0
Sleep
Output:' - A driver “IS-A” person (driver is a person): => "INHERITANCE".
' - A driver “HAS-A” driver's license (driver license only existing for the driver): => "COMPOSITION".
' - A driver “USES-A” vehicle (vehicle lifetime independent of the driver life): => "AGGREGATION".
Type Person
Public:
Dim As String full_name
Declare Constructor (ByRef _full_name As String)
Protected: '' to forbid at compile time the default-construction attempt of a Person instance
Declare Constructor ()
End Type
Constructor Person (ByRef _full_name As String)
This.full_name = _full_name
End Constructor
Type Driver_license
Public:
Dim As Integer number
End Type
Type Vehicle
Public:
Dim As String registration
Declare Constructor (ByRef _registration As String)
End Type
Constructor Vehicle (ByRef _registration As String)
This.registration = _registration
End Constructor
Type Driver Extends Person '' inheritance
Public:
Dim As Driver_license dl '' composition
Dim As Vehicle Ptr pv '' aggregation
Declare Constructor (ByRef _full_name As String, ByRef _dl As Driver_license)
End Type
Constructor Driver (ByRef _full_name As String, ByRef _dl As Driver_license)
Base(_full_name)
This.dl = _dl
End Constructor
Dim As Driver d1 = Driver("User fxm", Type<Driver_license>(123456789))
Dim As Vehicle Ptr pv1 = New Vehicle("ABCDEFGHI")
d1.pv = pv1
Print "Person full name : " & d1.full_name
Print "Driver license number : " & d1.dl.number
Print "Vehicle registration : " & d1.pv->registration
Delete pv1
d1.pv = 0
Sleep
Person full name : User fxm Driver license number : 123456789 Vehicle registration : ABCDEFGHI
Back to Programmer's Guide