应用场景:
一个人身高175cm,他觉得太矮泡不到妹子,穿了一双加厚的袜子增加1cm,嗯感觉效果不明显,再踩在一双加厚1cm的鞋垫,最后还不满意,再穿了一双内置增高3cm的鞋子,成功地把自己的身高“提高”到180cm+,他就被装饰成“大长腿”了。
什么是装饰者模式?
装饰者模式简单来说就是你想拓展A的功能,但是又不想把它拆了重建,这时你可以创建B,模仿(装饰)A的所有功能,然后在某些方面上模仿(装饰)得比A还要复杂,最后取而代之,以后要拓展B再创建C、D、E,一层套一层,理论上可以无限地嵌套。
在不需要修改原对象代码的基础上创建新的对象(装饰类),动态地把原对象包囊(嵌套),再拓展新的方法。
在学习I/O流时,可能很多新手会奇怪为什么new DataInputStream(new InputStream),是因为使用了装饰者模式
类图:
MSN、ICQ为被装饰类,talk()方法代码能进行普通文字聊天。
QQ、WeChat为装饰类,talk()方法代码除了基本聊天还可以显示用户名称和消息发送日期。以后还能创建新类,加上视频聊天、语音通话。
装饰者和被装饰者继承同一父类,所以装饰者可以替代并且扩展父类的行为
它们的功能不断地在原基础上加强。
talk()方法嵌套(递归)。
代码:
public abstract class Contact {
String name = "未知App";
public abstract void talk();
public String getName() {
return name;
};
}
public abstract class Disguise extends Contact {
Contact contact;
public abstract String getName();
}
public class ICQ extends Contact {
public ICQ(String name) {
this.name = name;
}
@Override
public void talk() {
System.out.println("一般聊天,噢噢噢");
}
@Override
public String getName() {
return this.name;
}
}
public class QQ extends Disguise {
public QQ(String name,Contact contact) {
this.name = name;
this.contact = contact;
}
@Override
public String getName() {
return this.name + "装饰了" + contact.getName();
}
@Override
public void talk() {
System.out.println(name + " " + new Date().toString());
contact.talk();
System.out.println("装饰后输出");
}
}
public static void main(String[] args) {
Contact contact = new ICQ("ICQ");
contact.talk();
System.out.println("--------------分割线--------------");
QQ qq = new QQ("QQ",contact);
qq.talk();
}
输出结果:
值得一提的是装饰类由于继承了相同的父类,所以可以相互装饰使用,正如Head First 里所举的例子:咖啡可以不停地放调料。
总结
装饰者模式的优点是装饰类可以嵌套的方式来达到功能扩展的目的。理论上可以无限嵌套,但是实际上,当装饰若干次会发现,会产生许多的装饰对象,程序因此变得复杂。