Understanding the Single Responsibility Principle: How to Improve Code Maintainability and Scalability
The Single Responsibility Principle (SRP) is an important concept in object-oriented programming that states that a module, class, or function should execute only one particular task in a program. The SRP ensures that a class has a single responsibility, which is, in turn, also encapsulated by the class. The SRP is one of the essential SOLID principles introduced by Robert C. Martin in his book "Agile Software Development, Principles and Pattern."
The importance of the SRP lies in its ability to improve code maintainability and scalability. This article will dive deep into the SRP to build maintainable and scalable code.
Writing Maintainable And Scalable Code Using The SRP
As we write programs, we tend to get more concerned about the program's functionality than the codebase structure. Fast forward to the future, and we discover that the codebase looks unnecessarily bulky and confusing in our attempt to make changes or add a new feature to the program. Playing around with the program to know its state and where to add the feature, we discover that multiple methods with different tasks have been jam-packed into a class, making the latter difficult to work around. In this scenario, the discovery is due to adding a new feature. In others, it may not be until we are testing or debugging that we discover the shoddy code we have written. While these programs may function properly in some cases, the problem arises when there is a need to make modifications later in the future in others. The single responsibility principle, if followed, prevents such scenarios and helps us write better, more maintainable, and more scalable code.
The SRP's strength lies in improving code maintainability and scalability. Observing classes in a well-written program reveals that the classes have a single, well-defined responsibility. In these programs, the programmers can easily add new features or changes without introducing bugs or breaking existing functionality. Additionally, by applying the SRP to our codes, we design more modular systems, which are easier to scale and adapt to changing requirements.
SRP stresses the importance of holding a class, module, or function to a single responsibility. As the popular SRP phrase by Robert Martin goes, "a class should have only one reason to change," we should write classes that address a single task and solve it. However, this does not mean we should oversimplify our programs by writing several classes with just one method; instead, we can have all the methods working together to achieve a specific function in a single class. For example, a class responsible for handling user input should only contain methods for handling user input and should not comprise methods for handling database queries or sending emails.
SRP In Action
To understand better, let’s look at an example of SRP’s implementation in an e-commerce application order processing codebase.
class OrderProcessor {
constructor(orderId, customerId, productId, quantity) {
this.orderId = orderId;
this.customerId = customerId;
this.productId = productId;
this.quantity = quantity;
}
validateOrder() {
// validate the order
}
calculateTotal() {
// calculate the order's total cost
}
processPayment() {
// process order payment
}
sendConfirmation() {
// send confirmation email to text to the customer
}
}
The OrderProcessor
class has multiple methods here. Before we conclude, let’s run a quick analysis.
The class has four methods:
validate_order: For validating the order of a customer
calculate_total: For calculating the total worth of the purchased items
process_payment: To facilitate the payment process. It includes validating the credit card and making withdrawals for the items purchased.
send_confirmation: If the credit card is successfully validated and the price of goods bought is withdrawn, the method sends a confirmation message through email or text to the customer or even lashes out a success alert on the screen. On the other hand, if the payment process is unsuccessful, the customer receives an error message.
These four methods handle different tasks in the OrderProcessor
class, respectively and have multiple reasons to change. Therefore, they should be separated.
Refactoring the code using the SRP, we have this:
class Order {
constructor(orderId, customerId, productId, quantity) {
this.orderId = orderId;
this.customerId = customerId;
this.productId = productId;
this.quantity = quantity;
}
}
class OrderValidator {
validateOrder(order) {
// validate the order
}
}
class OrderCalculator {
calculateTotal(order) {
// Calculate the total cost of the order
}
}
class PaymentProcessor {
processPayment(order) {
// Process the payment for the order
}
}
class ConfirmationSender {
sendConfirmation(order) {
// Send a confirmation email to the customer
}
}
In this code refactor, each class has a single responsibility. The Order
class is responsible for holding the order's data. The OrderValidator
class validates the order alone. The OrderCalculator
class calculates the order's total cost alone. The PaymentProcessor
class processes the payment. Finally, the ConfirmationSender
class only sends a confirmation email or text. This refactoring adheres to the Single Responsibility Principle because each class has a single reason to change.
Real-world Applications Of SRP
We can apply the Single Responsibility Principle (SRP) in real-world scenarios to improve code maintainability and scalability. One good way to identify if the SRP is required in an application is to examine its component breakdown. If more than one unrelated task is performed in a component, you should know it’s time to incorporate SRP. Here are a few examples:
Implementing a Payment System: A payment system like the example in the SRP in Action section has several components, such as validating the payment details, processing the payment, and sending a confirmation email or text. We can implement each component as a separate class and assign it a single responsibility, making it easier to change or update one component without affecting the others.
Building a CRUD Application: A CRUD (create, read, update, delete) application, such as a blog, an e-commerce platform, or even a simple to-do app, should have many classes that handle different specific responsibilities. These responsibilities include handling the database, rendering views, and managing user input. By giving each class a responsibility, the app becomes easier to update (for example, by implementing authentication and authorization features), modify, or remove a feature without disrupting the entire application.
Developing a Microservices Architecture: Microservices are an architectural pattern allowing developers to structure an application based on offered services collectively. Following the microservices architecture, a huge chunk of an application is divided into separate entities, each handling a specific task. Each service is responsible for a single functionality, which makes the application easier to change, update, or debug.
Building Business Logic: Business logic handles information exchange operations between the user interface and the server. It has multiple flows, with each flow possessing its validation and action. By assigning each flow a responsibility, the business logic adheres to the SRP and becomes simpler to alter or update.
These are only a few examples of situations where we can use the SRP in the real world. While there are more examples, the major lesson here is that by adopting the SRP, we can build classes and systems that are easier to maintain and scale, making it simpler to add new features or make adjustments without tampering with existing functionality.
SRP Implementation Steps And Best Practices
We have learned what SRP is and reviewed some of its application examples; now, let's learn how to implement it and its best practices.
Here are 3 easy steps to implement the Single Responsibility Principle in your code:
Break down the program into components: This is the first important step to take in implementing the SRP. By breaking down our program into components, we have a clear representation of the system's logic and can assign each component a single task to handle.
Determine a class's responsibilities: Each component will have classes and functions performing one task or another. Looking closely at our class's methods or functions, we can figure out if it adheres to or violates the SRP.
Split a class with multiple responsibilities into multiple classes: If the class violates the SRP, we should create new classes and redistribute the larger class's tasks to them.
For further inspection, let's check out these few SRPs' best practices:
Keep classes small and focused: Let's endeavour to keep our classes small and focused on a single responsibility. This makes them easier to understand and maintain, but remember not to oversimplify things.
Avoid creating "God Classes": "God Classes" are classes with too many responsibilities and become difficult to maintain. If adhering to the SRP, we must look out for "God classes" in our codes and break them down into simple classes.
Be mindful of coupling: The SRP's application is limited to classes, methods, and functions. Coupling is the degree of interaction between classes, methods, or functions that violates the principle of private information. While coupling cannot be eliminated, it can be minimized. A method or function adhering to the SRP must have a single reason to change and should not be tightly coupled to another method or function. Also, minimizing coupling in our methods or functions makes them modular and easier to modify.
Test each responsibility individually: A class or method with multiple responsibilities is difficult to test. When we individually test the responsibilities of our classes or methods, the degree of difficulty tells us if they violate the SRP and should be split into smaller classes or methods with individual tasks.
Keep in mind the SOLID principle: The Single Responsibility Principle is the first of the SOLID principles, and it works hand in hand with other principles to improve the design and maintainability of our code. In the future, we'll explore the other SOLID principles with examples and real-world applications.
By following these best practices, you can create classes and systems that are modular, maintainable, and scalable, making it easier to add new features or make changes without introducing bugs or breaking existing functionality.
Conclusion
Adhering to the SRP is crucial to having well-designed, maintainable, scalable software. It makes our codebase easier to understand, less prone to errors, and more flexible to change. By splitting the responsibilities into different classes, methods, or even microservices, we can rest assured that any modification or addition to the system will have less impact on the existing functionality.
In this article, we have explored the single responsibility principle in depth, looked at some practical examples and real-world applications, and finally learned the best practices of SRP implementation. To be a better developer, we’ll have to go beyond the SRP into more advanced concepts of the SOLID principle and not only learn but also practice them.
For further learning, some resources to consider include:
"Agile Software Development, Principles, Patterns, and Practices" by Robert C. Martin
"Clean Code: A Handbook of Agile Software Craftsmanship" by Robert C. Martin
"Design Patterns: Elements of Reusable Object-Oriented Software" by Erich Gamma, Richard Helm, Ralph Johnson, and John Vlissides.
"Refactoring: Improving the Design of Existing Code" by Martin Fowler
These resources provide a more in-depth look at the SRP and other SOLID principles and offer guidance on their application in practice.
Watch out for my next article as I'll apply the Single Responsibility Principle in building a To-do app.
If you like my content, connect with me on Twitter @theDocWhoCodes
Gracias, Ciao, 👋.