SOLID Principles
SOLID is an acronym for five object-oriented design principes (OOP) by Robert C. Martin. Adopting this practices helps in reducing code smell, refactoring of code, etc.
- S - Single Responsibility
- O - Open-Closed
- L - Liskov Substitution
- I - Interface Segregation
- D - Dependency Inversion
- Single Responsibility
- Open Closed
- Liskov Substitution
- Interface Segregation
- Dependency Inversion
A class should have only one reason to change.
This means that a class should have only one responsibility or purpose.
β Bad Example (Violates SRP)
This Employee class handles multiple responsibilities (employee details & salary calculations).
class Employee {
private String name;
private double salary;
public Employee(String name, double salary) {
this.name = name;
this.salary = salary;
}
public void printDetails() {
System.out.println("Name: " + name + ", Salary: " + salary);
}
public double calculateBonus() {
return salary * 0.10;
}
}
π΄ Problem
- The class has two reasons to change:
- If employee details change.
- If salary calculation logic changes.
- Breaks SRP, making it harder to maintain.
β Good Example (Follows SRP)
We separate concerns into two classes.
class Employee {
private String name;
private double salary;
public Employee(String name, double salary) {
this.name = name;
this.salary = salary;
}
public String getName() {
return name;
}
public double getSalary() {
return salary;
}
}
class SalaryCalculator {
public double calculateSalary(Employee employee) {
return employee.getSalary() * 0.1;
}
}
β Why is this better?
Employee
only manages employee data.SalaryCalculator
only handles salary calculations.- Each class has a single responsibility β Easier to maintain & extend.
A class should be open for extension but closed for modification.
This means that we should be able to add new functionality without changing existing code. This helps prevent breaking existing functionality and makes the system more maintainable.
β Bad Example (Violates Open/Closed Principle)
class DiscountCalculator {
public double calculateDiscount(String customerType, double amount) {
if (customerType.equals("Regular")) {
return amount * 0.1;
} else if (customerType.equals("Premium")) {
return amount * 0.2;
}
return 0;
}
}
π΄ Problem Everytime we add a new customer type, we have to modify existing code, which violates the open/closed principle.
β Good Example (Follows Open/Closed Principle)
We will use polymorphism to extend functionality without modifying existing code.
interface DiscountStrategy {
double applyDiscount(double amount);
}
class RegularDiscountStrategy implements DiscountStrategy {
public double applyDiscount(double amount) {
return amount * 0.1;
}
}
class PremiumDiscountStrategy implements DiscountStrategy {
public double applyDiscount(double amount) {
return amount * 0.2;
}
}
class DiscountCalculator {
public double calculateDiscount(DiscountStrategy strategy, double amount) {
return strategy.applyDiscount(amount);
}
}
public class Main {
public static void main(String[] args) {
DiscountCalculator calculator = new DiscountCalculator();
DiscountStrategy regularStrategy = new RegularDiscountStrategy();
DiscountStrategy premiumStrategy = new PremiumDiscountStrategy();
System.out.println("Regular discount: " + calculator.calculateDiscount(regularStrategy, amount));
System.out.println("Premium discount: " + calculator.calculateDiscount(premiumStrategy, amount));
}
}
β Why is this better?
- We can add new discount strategies (e.g.
GoldCustomerDiscount
) without modifyingDiscountCalculator
. - Follows Open/Closed Principle by making it open for extension but closed for modification.
- Improves maintainability and scalability.
Subtypes must be substitutable for their base types without altering the correctness of the program.
This means that if a class inherits from another, it should be able to replace its parent without breaking the behavior of the system.
β Bad Example (Violates LSP)
Let's say we have a Bird
class and a Penguin
class that extends it.
class Bird {
void fly() {
System.out.println("Bird is flying");
}
}
class Sparrow extends Bird {
// Inherits fly() method correctly
}
class Penguin extends Bird {
// Penguins cannot fly, but they inherit fly() method
}
π΄ Problem
If we try to substitute Penguin
for Bird
, it breaks the expected behavior because penguins cannot fly. This violates LSP.
β Good Example (Follows LSP)
To fix this, we separate flying and non-flying birds.
abstract class Bird {
abstract void eat();
}
interface Flyable {
void fly();
}
class Sparrow extends Bird implements Flyable {
void eat() {
System.out.println("Sparrow eats seeds.");
}
public void fly() {
System.out.println("Sparrow flies.");
}
}
class Penguin extends Bird {
void eat() {
System.out.println("Penguin eats fish.");
}
}
β Why is this better?
- Now,
Penguin
doesnβt have an incorrectfly()
method. - We can substitute any
Bird
without worrying about incorrect behavior. - Follows Liskov Substitution Principle, making the system more flexible and maintainable.
A class should not be forced to implement interfaces it does not use.
Don't force a class to implement methods it doesn't use, instead, create smaller, specific interfaces.
β Bad Example (Violates ISP)
Suppose we have a Worker interface that includes multiple responsibilities.
interface Worker {
void work();
void eat();
}
class Robot implements Worker {
public void work() {
System.out.println("Robot is working.");
}
public void eat() {
throw new UnsupportedOperationException("Robots do not eat");
}
}
π΄ Problem
- The Robot class is forced to implement eat(), even though it doesnβt need it.
- This violates ISP because the interface has unrelated responsibilities.
β Good Example (Follows ISP)
To fix this, we split the interface into smaller, more specific ones.
interface Worker {
void work();
}
interface Eatable {
void eat();
}
class Human implements Worker, Eatable {
public void work() {
System.out.println("Human is working.");
}
public void eat() {
System.out.println("Human is eating.");
}
}
class Robot implements Worker {
public void work() {
System.out.println("Robot is working.");
}
}
β Why is this better?
RobotWorker
only implementsWorkable
, so no unused methods.HumanWorker
can implement bothWorkable
andEatable
as needed.- Follows Interface Segregation Principle, making the design clean and maintainable.
Depend on abstractions, not on concrete classes.
High-level modules should depend on abstractions, not concrete implementations, to reduce coupling and increase flexibility.
β Bad Example (Violates DIP)
class Keyboard {
void type() {
System.out.println("Typing on the keyboard...");
}
}
class Mouse {
void click() {
System.out.println("Clicking the mouse...");
}
}
class WindowsComputer {
private Keyboard keyboard;
private Mouse mouse;
public WindowsComputer(Keyboard keyboard, Mouse mouse) {
this.keyboard = keyboard;
this.mouse = mouse;
}
public void useComputer() {
keyboard.type();
mouse.click();
}
}
π΄ Problem
WindowsComputer
is tightly coupled toKeyboard
andMouse
.- If we need a new input device (e.g.
Touchpad
), we must modifyWindowsComputer
.
β Good Example (Follows DIP)
interface InputDevice {
void input();
}
class Keyboard implements InputDevice {
public void input() {
System.out.println("Typing...");
}
}
class Mouse implements InputDevice {
public void input() {
System.out.println("Clicking...");
}
}
class Computer {
private InputDevice inputDevice;
public Computer(InputDevice inputDevice) {
this.inputDevice = inputDevice;
}
public void useInputDevice() {
inputDevice.input();
}
}
public class Main {
public static void main(String[] args) {
Computer computerWithKeyboard = new Computer(new Keyboard());
Computer computerWithMouse = new Computer(new Mouse());
computerWithKeyboard.useInputDevice();
computerWithMouse.useInputDevice();
}
}
β Why is this better?
- Computer depends on
InputDevice
(an abstraction) instead of concrete classes. - We can easily switch input devices (e.g.,
Touchpad
) without modifyingComputer
. - Follows Dependency Inversion Principle, making it flexible and extendable.