Open Closed Principle (OCP) states that,
Software entities (Classes, modules, functions) should be OPEN for EXTENSION, CLOSED for MODIFICATION.
Lets try to reflect on the above statement- software entities once written shouldn’t be modified to add new functionality, instead one has to extend the same to add new functionality.
In other words you don’t touch the existing modules thereby not disturbing the existing functionality, instead you extend the modules to implement the new requirement. So your code is less rigid and fragile and also extensible.
How can we confirm to OCP principle?
Its simple – Allow the modules (classes) to depend on the abstractions, there by new features can be added by creating new extensions of these abstractions.
Let me try to explain with an example:
Suppose you are writing a module to approve personal loans and before doing that you want to validate the personal information, code wise we can depict the situation as:
01 | public class LoanApprovalHandler |
03 | public void approveLoan(PersonalValidator validator) |
05 | if ( validator.isValid()) |
11 | public class PersonalLoanValidator |
13 | public boolean isValid() |
So far so good. As you all know the requirements are never the same and now its required to approve vehicle loans, consumer goods loans and what not. So one approach to solve this requirement is to:
01 | public class LoanApprovalHandler |
03 | public void approvePersonalLoan (PersonalLoanValidator validator) |
05 | if ( validator.isValid()) |
10 | public void approveVehicleLoan (VehicleLoanValidator validator ) |
12 | if ( validator.isValid()) |
19 | public class PersonalLoanValidator |
21 | public boolean isValid() |
26 | public class VehicleLoanValidator |
28 | public boolean isValid() |
We have edited the existing class to accomodate the new requirements- in the process we ended up changing the name of the existing method and also adding new methods for different types of loan approval. This clearly violates the OCP. Lets try to implement the requirement in a different way:
02 | * Abstract Validator class |
03 | * Extended to add different |
04 | * validators for different loan type |
06 | public abstract class Validator |
08 | public boolean isValid(); |
11 | * Personal loan validator |
13 | public class PersonalLoanValidator |
16 | public boolean isValid() |
Now using the above validators we can write a LoanApprovalHandler to use the Validator abstraction.
01 | public class LoanApprovalHandler |
03 | public void approveLoan(Validator validator) |
05 | if ( validator.isValid()) |
So to accommodate any type of loan validators we would just have create a subclass of Validator and then pass it to the approveLoan method. That way the class is CLOSED for modification but OPEN for extension.
Another example:
I was thinking of another hypothetical situation where the use of OCP principle can be of use. The situation is some thing like: “We maintain a list of students with their marks, unique identification(uid) and also name. Then we provide an option to get the percentage in the form of uid-percentage name value pairs.”
06 | public Student(String name, double percentage, int uid) |
09 | this .percentage = percentage; |
We collect the student list into a generic class:
02 | private List<Student> studentList; |
03 | public StudentBatch() { |
04 | studentList = new ArrayList<Student>(); |
06 | public void getSutdentMarkMap(Hashtable<Integer, Double> studentMarkMap) { |
07 | if (studentMarkMap == null ) { |
10 | for (Student student : studentList) { |
11 | studentMarkMap.put(student.uid, student.percentage); |
16 | * @param studentList the studentList to set |
18 | public void setStudentList(List<Student> studentList) { |
19 | this .studentList = studentList; |
Suppose we need to maintain the order of elements in the Map by their insertion order, so we would have to write a new method to get the map in the insertion order and for that we would be using LinkedHashMap. Instead if the method- getStudentMarkMap() was dependent on the Map interface and not the Hashtable concrete implementation, we could have avoided changing the StudentBatch class and instead pass in an instance of LinkedHashMap.
1 | public void getSutdentMarkMap(Map<Integer, Double> studentMarkMap) { |
2 | if (studentMarkMap == null ) { |
5 | for (Student student : studentList) { |
6 | studentMarkMap.put(student.uid, student.percentage); |
PS: I know that Hashtable is an obsolete collection and not encouraged to be use. But I thought this would make another useful example for OCP principle.
Some ways to keep your code closer to confirming OCP:
Making all member variables private so that the other parts of the code access them via the methods (getters) and not directly.
Avoiding typecasts at runtime- This makes the code fragile and dependent on the classes under consideration, which means any new class might require editing the method to accommodate the cast for the new class.
No comments:
Post a Comment