The Open/Closed Principle states that software entities (classes, modules, functions, etc.) should be open for extension but closed for modification. This means that we can add new functionality to existing code without altering its existing source code, thereby reducing the risk of introducing bugs. The gist is to keep your code stable while being able to implement new features when needed.
Let's break down how to implement the Open/Closed Principle in Python through practical examples.
Imagine a payment processing system that currently supports only credit card payments. Here’s how you'd typically start implementing it without adhering to the OCP:
class PaymentProcessor: def process_credit_card(self, amount): print(f"Processing credit card payment of ${amount}")
Now, if you wanted to add PayPal support, you'd have to modify the PaymentProcessor
class directly:
class PaymentProcessor: def process_credit_card(self, amount): print(f"Processing credit card payment of ${amount}") def process_paypal(self, amount): print(f"Processing PayPal payment of ${amount}")
This approach violates the Open/Closed Principle. Each time we want to add a new payment method, we must modify the existing class, which can lead to unintended side effects.
To implement OCP, we can define an interface for payment processing and create separate classes for each payment type that implements this interface:
from abc import ABC, abstractmethod class PaymentMethod(ABC): @abstractmethod def process_payment(self, amount): pass class CreditCardPayment(PaymentMethod): def process_payment(self, amount): print(f"Processing credit card payment of ${amount}") class PayPalPayment(PaymentMethod): def process_payment(self, amount): print(f"Processing PayPal payment of ${amount}") class PaymentProcessor: def __init__(self): self.payment_methods = [] def add_payment_method(self, payment_method: PaymentMethod): self.payment_methods.append(payment_method) def process_payment(self, amount): for method in self.payment_methods: method.process_payment(amount) # Usage processor = PaymentProcessor() processor.add_payment_method(CreditCardPayment()) processor.add_payment_method(PayPalPayment()) processor.process_payment(100)
Here, the PaymentProcessor
class remains untouched when new payment methods are added. We can create new classes (like BitcoinPayment
) to extend functionalities without altering existing code, thus adhering to OCP.
Imagine we need to compute areas of different shapes. Usually, you might start with something like this:
class AreaCalculator: def calculate_area(self, shape): if shape['type'] == 'circle': return 3.14 * shape['radius'] ** 2 elif shape['type'] == 'rectangle': return shape['width'] * shape['height']
Each time you need to add a new shape, you end up modifying the AreaCalculator
, which is again a violation of OCP.
Let’s implement OCP:
from abc import ABC, abstractmethod class Shape(ABC): @abstractmethod def area(self): pass class Circle(Shape): def __init__(self, radius): self.radius = radius def area(self): return 3.14 * self.radius ** 2 class Rectangle(Shape): def __init__(self, width, height): self.width = width self.height = height def area(self): return self.width * self.height class AreaCalculator: def calculate_area(self, shape: Shape): return shape.area() # Usage circle = Circle(5) rectangle = Rectangle(10, 5) calculator = AreaCalculator() print(calculator.calculate_area(circle)) # Output: 78.5 print(calculator.calculate_area(rectangle)) # Output: 50
Now, whenever you want to add a new shape, like Triangle
, you can simply create a new class extending Shape
, with its own area()
method, without touching the AreaCalculator
or any existing shape classes. This approach keeps your code clean, maintainable, and adheres to the Open/Closed Principle.
The Open/Closed Principle drives the design of flexible and maintainable software systems. By ensuring that our classes are open for extension but closed for modification, we can facilitate growth while maintaining stability. The key takeaway is to use abstraction and polymorphism to allow new functionalities to flourish without disrupting the existing codebase.
These principles are foundational in creating scalable applications in Python, and understanding them can significantly improve your software design capabilities. Happy coding!
10/02/2025 | Design Patterns
15/01/2025 | Design Patterns
06/09/2024 | Design Patterns
09/10/2024 | Design Patterns
12/10/2024 | Design Patterns
10/02/2025 | Design Patterns
10/02/2025 | Design Patterns
10/02/2025 | Design Patterns
12/10/2024 | Design Patterns
12/10/2024 | Design Patterns
12/10/2024 | Design Patterns
15/01/2025 | Design Patterns