在Java面向对象思想中,有一个重要的开闭原则,即类应该对扩展开放,对修改关闭,扩展就是添加新功能的意思,因此该原则要求在添加新功能时不需要修改代码。而符合开闭原则最典型的设计模式就是装饰者模式。
一、什么是装饰者模式?
装饰者模式可以在不改变代码结构的前提下,对已有的对象添加新的功能。
如果不考虑装饰者模式,我们要扩展一个类,可以使用继承的方式来实现,通过对扩展类添加子类,在子类中对扩展类的方法进行补充。通过这种继承的方式进行扩展有一个弊端,随着扩展功能的增多,子类的数量会增多。所以采用装饰者模式来替代继承对类进行扩展。
优点:装饰类和被装饰类可以独立发展,不会相互耦合。与继承的静态性不同,装饰者模式支持动态扩展和动态撤销。
装饰者模式类图:
二、案例代码
情景假设:如果我们拥有一个初代机器人,它只能实现扫地的功能,我们需要对它的功能进行扩展——增加一个洗碗的功能。
首先创建一个Robot接口:
interface Robot{
void doSomething();
}
创建初代机器人,实现扫地功能:
class FirstRobot implements Robot{
@Override
public void doSomething() {
System.out.println("扫地");
}
}
创建装饰类,对初代机器人增加洗碗功能:
class DecoratorRobot implements Robot{
private Robot robot;
public DecoratorRobot(Robot robot){ //通过含参构造方法将被装饰的对象传递进来
this.robot = robot;
}
@Override
public void doSomething() {
robot.doSomething();
}
public void doMorething(){ //扩展新功能
robot.doSomething();
System.out.println("洗碗");
}
}
主函数:
public class DecoratorDemo {
public static void main(String[] args) {
new DecoratorRobot(new FirstRobot()).doMorething();
}
}
执行结果:
扫地
洗碗
注意:在类图中我们常把装饰类设置为抽象类,通过定义抽象类的子类来实现更多的装饰器。
主函数中的装饰类和被装饰对象的调用方式,也很像IO流中的链接机制:
new DataInputStream(new BufferedInputStream(new FileInputStream()));
即IO流中就大量采用了装饰者模式来对类的功能进行增强。
三、装饰者模式和代理模式的区别?
装饰者模式中,装饰者和被装饰者都实现同一个接口;在代理模式中,代理类和被代理类也都实现同一个接口。两者好像都是实现对类的增强,但是具体有什么区别呢?
实际上装饰者模式和代理模式还是有很多区别的。
1.装饰者模式更注重在一个对象上动态地添加方法,而代理模式关注于控制对对象的访问。
2.使用代理模式的时候,我们常常在一个代理类中创建一个对象的实例,而在装饰者模式中,我们通常的做法是将原始对象当做一个参数传递给装饰类的构造器。