Liskov Substitution Principle (LSP):

SOLID - SRP

The Liskov Substitution Principle (LSP) states that objects of a superclass should be substitutable for objects of its subclasses without affecting the correctness of the program. In other words, if a program is designed to work with a certain type or interface, substituting it with any of its derived types should not cause any unexpected behavior or violate the expected behavior.

Let's consider a practical use case to understand the Liskov Substitution Principle:

Use Case: Animal Feeding System

Example: Suppose we are developing an animal feeding system that allows feeding different types of animals, such as dogs, cats, and birds. Each animal has a specific way of eating.

Without applying the LSP:

In an initial implementation, we might have a single Animal class with an eat() method that contains the eating behavior for all types of animals. However, this violates the LSP because substituting one animal type for another could result in incorrect or unexpected behavior.

With Applying the LSP:

To apply the LSP, we can use inheritance and polymorphism to create a class hierarchy that adheres to the LSP:

Syntax:
                
    public abstract class Animal {
        public abstract void eat();
    }

    public class Dog extends Animal {
        @Override
        public void eat() {
            // eating behavior for dogs
        }
    }

    public class Cat extends Animal {
        @Override
        public void eat() {
            // eating behavior for cats
        }
    }

    public class Bird extends Animal {
        @Override
        public void eat() {
            // eating behavior for birds
        }
    }

    public class FeedingSystem {
        public void feedAnimal(Animal animal) {
            animal.eat();
        }
    }
                
            

In this example, we have an abstract Animal class representing the common behavior of all animals. Each specific animal type (Dog, Cat, Bird) extends the Animal class and provides its own implementation of the eat() method.

The FeedingSystem class has a feedAnimal() method that takes an Animal object as a parameter and calls its eat() method. Since all animal types inherit from the Animal class, they can be substituted and passed as arguments to the feedAnimal() method without violating the LSP.

By following the LSP, the animal feeding system can handle different types of animals uniformly. The system doesn't need to know the specific type of animal it's feeding; it only relies on the Animal interface or base class. This allows for easier extension of the system with new animal types, as they can be seamlessly substituted without affecting the correctness of the feeding process.

Overall, adhering to the Liskov Substitution Principle promotes code reuse, modularity, and the ability to work with a variety of derived types interchangeably while maintaining the expected behavior of the program.