SOLID Principals - Open-Closed Principle (OCP)
The Open-Closed Principle (OCP) is one of the SOLID principles that emphasizes the idea that classes should be open for extension but closed for modification. This principle encourages developers to design software entities in a way that allows for new functionality to be added without altering existing code.
Understanding the Open-Closed Principle
The Open-Closed Principle states that a class should be open for extension but closed for modification. This means that you should be able to extend the behavior of a class without changing its source code. The goal is to promote software entities that are easily extensible without introducing potential bugs or unintended side effects.
Example 1: Basic Implementation
Let’s consider a simple example where we have a class representing different shapes, and we want to calculate the area of each shape. Initially, we may have a class like this:
class Rectangle
attr_accessor :width, :height
def initialize(width, height)
@width = width
@height = height
end
def area
width * height
end
end
In this example, if we later want to add support for calculating the area of a circle, we might be tempted to modify the existing Rectangle
class. However, this violates the Open-Closed Principle.
Refactoring for Open-Closed Principle
To adhere to the Open-Closed Principle, we can introduce an abstract base class (Shape
) and create specific implementations for each shape:
class Shape
# Common methods or properties for all shapes can go here
end
class Rectangle < Shape
attr_accessor :width, :height
def initialize(width, height)
@width = width
@height = height
end
def area
width * height
end
end
class Circle < Shape
attr_accessor :radius
def initialize(radius)
@radius = radius
end
def area
Math::PI * radius**2
end
end
Now, we can extend our system by adding new shapes without modifying the existing code. Each new shape is a class that inherits from the common Shape
class and provides its own implementation of the area
method.
Example 2: Strategy Pattern
The Open-Closed Principle can also be applied using design patterns. Let’s consider the Strategy Pattern, which allows us to define a family of algorithms, encapsulate each one, and make them interchangeable.
class Order
attr_accessor :total, :discount_strategy
def initialize(total, discount_strategy)
@total = total
@discount_strategy = discount_strategy
end
def final_total
total - discount_strategy.apply_discount(total)
end
end
class ChristmasDiscount
def apply_discount(total)
total * 0.1 # 10% discount for Christmas
end
end
class BlackFridayDiscount
def apply_discount(total)
total * 0.2 # 20% discount for Black Friday
end
end
In this example, the Order
class is open for extension because we can add new discount strategies without modifying the existing code. Each discount strategy is encapsulated in its own class, adhering to the Open-Closed Principle.
The Open-Closed Principle encourages a design philosophy that facilitates extending software entities without modifying their existing code. By using abstraction, inheritance, and design patterns, developers can create systems that are more flexible, maintainable, and scalable.