The Open-Closed Principle: Designing for Extension and Stability

Kittipat.Po
4 min readJul 28, 2024

--

This article is part of a series that explores the SOLID principles of object-oriented design, providing insights and practical examples to help you write better code. The content and principles discussed are derived from the book “Clean Architecture” by Robert C. Martin, offering guidance on creating maintainable, scalable, and robust software systems.

Table of Contents

  1. The Single Responsibility Principle
  2. The Open-Closed Principle
  3. The Liskov Substitution Principle 🚧
  4. The Interface Segregation Principle 🚧
  5. The Dependency Inversion Principle 🚧

Introduction

The Open-Closed Principle (OCP) is a cornerstone of software design that was introduced by Bertrand Meyer in 1988. This principle states:

A software artifact should be open for extension but closed for modification.

In essence, OCP advocates that the behavior of a software component should be extendable without altering its existing code. This principle is fundamental for achieving robust and flexible software architecture.

Understanding the Open-Closed Principle

The Open-Closed Principle emphasizes two key aspects:

  1. Open for Extension: New functionality can be added to the system when requirements change.
  2. Closed for Modification: Existing code remains unchanged, preventing the introduction of bugs in previously working code.

Applying OCP effectively helps to reduce the impact of changes and ensures that new features can be added with minimal disruption.

Example: Financial Report System

Let’s consider a scenario to illustrate the application of OCP. Imagine we have a system that displays financial summaries on a web page. The stakeholders now request that this information be printed as a report with specific formatting, including pagination and special handling for negative numbers.

Initial Design

Initially, we might have a simple class to generate the web-based financial report:

This design works well for the web report but fails to accommodate the new printing requirements without modifying the existing FinancialReport class.

Applying OCP

To adhere to OCP, we should separate the responsibilities of generating the report data and presenting it. We can create an interface for the report generation and two concrete implementations for HTML and print formats:

Now, the FinancialReport class can use these generators without needing to be modified:

By introducing the ReportGenerator interface and its implementations, we can extend the reporting functionality without altering the existing code, thus adhering to the OCP.

Architectural Considerations

When applying OCP at the architectural level, we must carefully manage dependencies and isolate changes. The goal is to ensure that modifications in one part of the system do not propagate unintended changes in others.

Component Design

Consider a system with components for processing data, interacting with a database, and presenting data. By properly structuring these components and their dependencies, we can protect core business logic from changes in peripheral parts:

In this structure:

  • The DataProcessor handles the core business logic and should be protected from changes in DataRepository, ReportGenerator, and Formatter.
  • Dependencies flow inward toward the core components, shielding them from changes in outer components.

Here’s an example in Go:

  • DataRepository fetches the data.
  • ReportGenerator generates the report.
  • DataProcessor coordinates these actions without being tightly coupled to specific implementations.

Conclusion

The Open-Closed Principle is essential for building extensible and maintainable software systems. By designing components that are open for extension but closed for modification, we can introduce new functionality with minimal risk of disrupting existing code. Adhering to OCP not only improves code quality but also enhances the overall stability and scalability of the software architecture.

As you design and refactor your software, keep the Open-Closed Principle in mind to create systems that are both flexible and resilient to change.

--

--