Post

CS - 3 Association Between Classes

Pulled from: Mosh Hamedani
Programming with Mosh - Learn the Skills to Land Your Dream Job



Class Coupling

A measure of how interconnected classes and subsystems are. The more coupled classes, the harder it is to change them. A change in one class may affect many other classes. Loosely coupled software, as opposed to tightly coupled software, is easier to change.

Two types of relationships between classes: Inheritance and Composition.


Inheritance

A kind of relationship between two classes that allows one to inherit code from the other.

  • Referred to as Is-A relationship: A Car is a Vehicle
  • Benefits: code re-use and polymorphic behaviour.
1
2
3
4
public class Car : Vehicle 
{ 
    // Members
}


Composition

A kind of relationship that allows one class to contain another.

  • Referred to as Has-A relationship: A Car has an Engine 1
  • Benefits: Code re-use, flexibility and a means to loose-coupling
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
// Base
public class Animal
{
    public int Age { get; set; }

    public void Eat() 
    {
        Console.WriteLine("Nom Nom");
    }
    public void Breath() 
    {
        Console.WriteLine("Snore snore");
    }
}

// derived with composition
public class Dog
{
    private readonly Animal _Animal;
    
    public Dog()
    {
    
    }
    public Dog(Animal animal)
    {
        _Animal = animal;
    }
    
    public void Eat() 
    {
        _Animal.Eat();
    }
    public void Breath() 
    { 
        _Animal.Breath();
    }
}




Favour Composition over Inheritance

  • Problems with inheritance:
    • Easily abused by amateur designers / developers
    • Leads to large complex hierarchies
    • Such hierarchies are very fragile and a change may affect many classes
    • Results in tight coupling
  • Benefits of composition:
    • Flexible
    • Leads to loose coupling
  • Having said all that, it doesn’t mean inheritance should be avoided at all times. In fact, it’s great to use inheritance when dealing with very stable classes on top of small hierarchies. As the hierarchy grows (or variations of classes increase), the hierarchy, however, becomes fragile. And that’s where composition can give you a better design.




Composition variants:

Lets look at 2 slightly different means of composition:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
// Default variant
public class Dog
{
    private readonly Animal _Animal;
    
    public Dog()
    {
    
    }
    public Dog(Animal animal)
    {
        _Animal = animal;
    }
    
    public void Eat() 
    {
        _Animal.Eat();
    }
    public void Breath() 
    { 
        _Animal.Breath();
    }
}


1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
// Variant 2
public class Dog 
{ 
    private readonly Animal _Animal; 
    
    public Dog() 
    { 
        _Animal = new Animal(); 
    } 
    
    public void Eat() 
    { 
        _Animal.Eat(); 
    } 
    
    public void Breath() 
    { 
        _Animal.Breath(); 
    } 
}

In the first example, the Dog class has a “has-a” relationship with the Animal class. This means that a Dog object has an instance of the Animal class as a member variable. The Dog class relies on the Animal class to provide some of its functionality, specifically the Eat() and Breath() methods, which are called through the _Animal member variable.

In the second example, the Dog class creates a new instance of the Animal class within its constructor. This means that a Dog object has its own Animal object as a member variable, rather than relying on an existing Animal object that was passed to the constructor.

The main difference between these two approaches is in how the Animal object is created and managed. In the first example, the Animal object is created externally and passed to the Dog constructor, which allows for more flexibility in how the Dog object is created and used. For example, you could create multiple Dog objects that all share the same Animal object. This can be useful in some situations, but it can also make the code more complex.

In the second example, the Animal object is created internally by the Dog constructor, which simplifies the code somewhat. However, this approach is less flexible, since each Dog object has its own Animal object and there is no way to share an Animal object between multiple Dog objects.

So, which approach to use depends on the specific needs of your program. If you need more flexibility and want to be able to share an Animal object between multiple Dog objects, then the first approach is probably better. If you just need a simple Dog object that always has its own Animal object, then the second approach may be sufficient.

This post is licensed under CC BY 4.0 by the author.