开闭原则
所谓的开闭原则(Open-Closed Principle,缩写为OCP),是指对扩展开放,对修改关闭。在程序需要扩展的时候,不能去修改原有的代码,而是实现一个热插拔的效果。
开闭原则是为了使程序的扩展性好,易于维护和升级。
要想达到这样的效果,我们需要使用接口和抽象,而软件中易变的细节可以从抽象类派生出来的实现类来扩展。当软件需要变化时,只需要从抽象类派生一个实现类来扩展就可以了。
示例
下面用显示主题来模拟下开闭原则:
定义了一个抽象显示主题类AbstractDisplayTheme
package com.test.principle;
/**
* 抽象类
*/
public abstract class AbstractDisplayTheme {
public abstract void display();
}
分别定义两个具体实现类DefaultDisplayTheme、LightDisplayTheme来实现抽象类中的方法
package com.test.principle;
/**
* 默认显示主题
*/
public class DefaultDisplayTheme extends AbstractDisplayTheme {
@Override
public void display() {
System.out.println("这是默认显示主题");
}
}
package com.test.principle;
/**
* 高亮显示主题
*/
public class LightDisplayTheme extends AbstractDisplayTheme {
@Override
public void display() {
System.out.println("这是高亮显示主题");
}
}
定义一个聚合类CustomInput,来聚合显示主题类
package com.test.principle;
/**
* 可以用来更换不同的主题
*/
public class CustomInput {
private AbstractDisplayTheme displayTheme;
public void setDisplayTheme(AbstractDisplayTheme theme) {
this.displayTheme = theme;
}
public void display() {
displayTheme.display();
}
}
定义一个客户端类,模拟使用
开始使用默认主题:
package com.test.principle;
/**
*
*/
public class Client {
public static void main(String[] args) {
CustomInput customInput = new CustomInput();
// 使用默认主题
AbstractDisplayTheme theme = new DefaultDisplayTheme();
customInput.setDisplayTheme(theme);
customInput.display();
}
}
运行输出:
如果想改为高亮主题:
package com.test.principle;
/**
*
*/
public class Client {
public static void main(String[] args) {
CustomInput customInput = new CustomInput();
AbstractDisplayTheme theme = new LightDisplayTheme();
customInput.setDisplayTheme(theme);
customInput.display();
}
}
运行输出:
扩展另外一个绿色主题:对扩展开放
现在要扩展另外一个显示主题,绿色主题,那么只要重新定义一个类,来实现抽象主题类即可,这就是所谓的对扩展开放:
package com.test.principle;
/**
* 绿色主题显示类
*/
public class GreenDisplayTheme extends AbstractDisplayTheme {
@Override
public void display() {
System.out.println("这是绿色主题");
}
}
客户端代码修改一行,就可以使用绿色主题了:
package com.test.principle;
/**
*
*/
public class Client {
public static void main(String[] args) {
CustomInput customInput = new CustomInput();
AbstractDisplayTheme theme = new GreenDisplayTheme();
customInput.setDisplayTheme(theme);
customInput.display();
}
}
运行输出: