动态的给某些对象添加额外的功能
装饰模式:动态的给某些对象添加额外的功能 参考:
装饰模式也叫装饰器模式,python中的装饰器就是这种模式的体现,对于一个类,如果要添加一个新功能,除了修改代码外(违反开闭原则),可以使用继承,但通过继承添加新功能并不适合所有场景,如
装饰模式中的对象包括:
客户端如果希望给某个对象动态添加一个新功能,就可以把这个对象(被装饰对象)传递给装饰器,由装饰器实现新功能,并保存一个被装饰对象的引用,并返回给客户端一个装饰器对象,这样,被装饰对象原来的行为和属性并没有改变,甚至被装饰对象本身就没有改变,只是在外面套了一个壳子,新功能是这个壳子提供的。就像TCP/IP协议栈中,应用层的数据包到传输层通过加TCP或UDP首部来传输一样。
优点:
缺点:
比如卖烤冷面,最基本的就是面(抽象被装饰对象)具体的就是烤冷面(被装饰对象),然后可以往面里面加各种配料(抽象装饰器),如鸡蛋,辣条等(具体装饰器),由于不同配料的加入顺序对最后的烤冷面有影响,所以如果要用继承拓展“烤冷面”,那先加鸡蛋再加辣条和先加辣条再加鸡蛋就需要写两个子类,造成冗余重复,这种场景就适合适用装饰模式。
抽象被装饰对象
package pers.junebao.decorator_pattern;
public abstract class Noodles {
public String rawMaterial; // 配料
public abstract void sayWhoAmI();
}
具体的被装饰对象:
package pers.junebao.decorator_pattern;
public class BakedColdNoodles extends Noodles {
BakedColdNoodles() {
this.rawMaterial = "面"; // 最原始的烤冷面,配料只有面
}
@Override
public void sayWhoAmI() {
System.out.println("我是普通烤冷面!");
}
}
抽象装饰器:
package pers.junebao.decorator_pattern.decorator;
import pers.junebao.decorator_pattern.Noodles;
public abstract class Burden extends Noodles {
public Noodles noodles; // 装饰器中保留一份被装饰对象的引用,方便客户端使用
public Burden(Noodles noodles) {
this.noodles = noodles;
}
}
具体的装饰器类:
加鸡蛋
package pers.junebao.decorator_pattern.decorator;
import pers.junebao.decorator_pattern.Noodles;
public class AddEggs extends Burden {
public AddEggs(Noodles noodles) {
super(noodles);
this.rawMaterial = noodles.rawMaterial + ", 鸡蛋";
}
@Override
public void sayWhoAmI() {
System.out.println("我是加了鸡蛋的烤冷面!!");
}
}
加辣条
package pers.junebao.decorator_pattern.decorator;
import pers.junebao.decorator_pattern.Noodles;
public class AddSpicyStrips extends Burden{
public AddSpicyStrips(Noodles noodles) {
super(noodles);
this.rawMaterial = noodles.rawMaterial + " ,辣条";
}
@Override
public void sayWhoAmI() {
System.out.println("我是加了辣条的烤冷面!!");
}
}
客户端:
package pers.junebao.decorator_pattern;
import pers.junebao.decorator_pattern.decorator.AddEggs;
import pers.junebao.decorator_pattern.decorator.AddSpicyStrips;
public class Main {
public static void main(String[] args) {
Noodles bcn = new BakedColdNoodles();
Noodles bcnAddEgg = new AddEggs(bcn);
bcnAddEgg.sayWhoAmI();
System.out.println(bcnAddEgg.rawMaterial);
Noodles bcnEggSpicyS = new AddSpicyStrips(bcnAddEgg);
bcnEggSpicyS.sayWhoAmI();
System.out.println(bcnEggSpicyS.rawMaterial);
}
}
/*
我是加了鸡蛋的烤冷面!!
面, 鸡蛋
我是加了辣条的烤冷面!!
面, 鸡蛋 ,辣条
*/
这样如果想先加辣条在家鸡蛋,就可以使用AddSpicyStrips先装饰BakedColdNoodles,再用AddEggs装饰AddSpicyStrips。