装饰器模式
装饰器模式(Decorator Pattern)允许向一个现有的对象添加新的功能,同时又不改变其结构。这种类型的设计模式属于结构型模式,它是作为现有的类的一个包装。
装饰器模式通过将对象包装在装饰器类中,以便动态地修改其行为。
这种模式创建了一个装饰类,用来包装原有的类,并在保持类方法签名完整性的前提下,提供了额外的功能。
我们通过下面的实例来演示装饰器模式的用法。其中,我们将把一个形状装饰上不同的颜色,同时又不改变形状类。
概要
意图
动态地给一个对象添加额外的职责,同时不改变其结构。装饰器模式提供了一种灵活的替代继承方式来扩展功能。
主要解决的问题
- 避免通过继承引入静态特征,特别是在子类数量急剧膨胀的情况下。
- 允许在运行时动态地添加或修改对象的功能。
使用场景
- 当需要在不增加大量子类的情况下扩展类的功能。
- 当需要动态地添加或撤销对象的功能。
实现方式
- 定义组件接口:创建一个接口,规定可以动态添加职责的对象的标准。
- 创建具体组件:实现该接口的具体类,提供基本功能。
- 创建抽象装饰者:实现同样的接口,持有一个组件接口的引用,可以在任何时候动态地添加功能。
- 创建具体装饰者:扩展抽象装饰者,添加额外的职责。
关键代码
- Component接口:定义了可以被装饰的对象的标准。
- ConcreteComponent类:实现Component接口的具体类。
- Decorator抽象类:实现Component接口,并包含一个Component接口的引用。
- ConcreteDecorator类:扩展Decorator类,添加额外的功能。
应用实例
- 孙悟空的72变:孙悟空(ConcreteComponent)通过变化(Decorator)获得新的能力。
- 画框装饰画:一幅画(ConcreteComponent)可以通过添加玻璃(ConcreteDecorator)和画框(ConcreteDecorator)来增强其展示效果。
优点
- 低耦合:装饰类和被装饰类可以独立变化,互不影响。
- 灵活性:可以动态地添加或撤销功能。
- 替代继承:提供了一种继承之外的扩展对象功能的方式。
缺点
- 复杂性:多层装饰可能导致系统复杂性增加。
使用建议
- 在需要动态扩展功能时,考虑使用装饰器模式。
- 保持装饰者和具体组件的接口一致,以确保灵活性。
注意事项
- 装饰器模式可以替代继承,但应谨慎使用,避免过度装饰导致系统复杂。
结构
装饰器模式包含以下几个核心角色:
- 抽象组件(Component):定义了原始对象和装饰器对象的公共接口或抽象类,可以是具体组件类的父类或接口。
- 具体组件(Concrete Component):是被装饰的原始对象,它定义了需要添加新功能的对象。
- 抽象装饰器(Decorator):继承自抽象组件,它包含了一个抽象组件对象,并定义了与抽象组件相同的接口,同时可以通过组合方式持有其他装饰器对象。
- 具体装饰器(Concrete Decorator):实现了抽象装饰器的接口,负责向抽象组件添加新的功能。具体装饰器通常会在调用原始对象的方法之前或之后执行自己的操作。
装饰器模式通过嵌套包装多个装饰器对象,可以实现多层次的功能增强。每个具体装饰器类都可以选择性地增加新的功能,同时保持对象接口的一致性。
实现
我们将创建一个 Shape 接口和实现了 Shape 接口的实体类。然后我们创建一个实现了 Shape 接口的抽象装饰类 ShapeDecorator,并把 Shape 对象作为它的实例变量。
RedShapeDecorator 是实现了 ShapeDecorator 的实体类。
DecoratorPatternDemo 类使用 RedShapeDecorator 来装饰 Shape 对象。
步骤 1
创建一个接口:
Shape.java
public interface Shape {
void draw();
}
步骤 2
创建实现接口的实体类。
Rectangle.java
public class Rectangle implements Shape {
@Override
public void draw() {
System.out.println("Shape: Rectangle");
}
}
Circle.java
public class Circle implements Shape {
@Override
public void draw() {
System.out.println("Shape: Circle");
}
}
步骤 3
创建实现了 Shape 接口的抽象装饰类。
ShapeDecorator.java
public abstract class ShapeDecorator implements Shape {
protected Shape decoratedShape;
public ShapeDecorator(Shape decoratedShape){
this.decoratedShape = decoratedShape;
}
public void draw(){
decoratedShape.draw();
}
}
步骤 4
创建扩展了 ShapeDecorator 类的实体装饰类。
RedShapeDecorator.java
public class RedShapeDecorator extends ShapeDecorator {
public RedShapeDecorator(Shape decoratedShape) {
super(decoratedShape);
}
@Override
public void draw() {
decoratedShape.draw();
setRedBorder(decoratedShape);
}
private void setRedBorder(Shape decoratedShape){
System.out.println("Border Color: Red");
}
}
步骤 5
使用 RedShapeDecorator 来装饰 Shape 对象。
DecoratorPatternDemo.java
public class DecoratorPatternDemo {
public static void main(String[] args) {
Shape circle = new Circle();
ShapeDecorator redCircle = new RedShapeDecorator(new Circle());
ShapeDecorator redRectangle = new RedShapeDecorator(new Rectangle());
//Shape redCircle = new RedShapeDecorator(new Circle());
//Shape redRectangle = new RedShapeDecorator(new Rectangle());
System.out.println("Circle with normal border");
circle.draw();
System.out.println("\nCircle of red border");
redCircle.draw();
System.out.println("\nRectangle of red border");
redRectangle.draw();
}
}
步骤 6
执行程序,输出结果:
Circle with normal border Shape: Circle Circle of red border Shape: Circle Border Color: Red Rectangle of red border Shape: Rectangle Border Color: Red
周霆
598***[email protected]
一个更易理解的实例:
装饰模式为已有类动态附加额外的功能就像LOL、王者荣耀等类Dota游戏中,英雄升级一样。每次英雄升级都会附加一个额外技能点学习技能。具体的英雄就是ConcreteComponent,技能栏就是装饰器Decorator,每个技能就是ConcreteDecorator;
输出:
周霆
598***[email protected]
该用户昵称违规
815***[email protected]
游戏里英雄皮肤的实现 是不是也比较适合装饰器模式
该用户昵称违规
815***[email protected]
郭为宇
gos***[email protected]
《HeadFirst 设计模式》里的装饰器模式的代码。
测试:
输出:
郭为宇
gos***[email protected]
这不是小明
xia***[email protected]
复杂些的实例
在《绝地求生:刺激战场》游戏里面我们都知道。
枪具有开火功能:
装饰上弹匣变更枪开火功能:
测试:输出:
现在我要装上4倍镜,使它具有4倍瞄准功能,这是枪支原本没有的功能。
扩展枪支接口功能:
测试:
输出:
现在我要装上8倍镜,它具有4倍瞄准功能,也具有8倍瞄准功能。
扩展接口:
测试:
输出:
进入4倍瞄准模式 [开炮!] 砰*10这不是小明
xia***[email protected]
Lonnie
354***[email protected]
Swift 实现:
Lonnie
354***[email protected]
Siskin.xu
sis***@sohu.com
Python 方式:
Siskin.xu
sis***@sohu.com
lz
luz***[email protected]
C++语言
控制台输出:
注意! 修饰器的顺序很重要,如果错误,则时间戳里的22字符创也会被屏蔽!
lz
luz***[email protected]
TooHandsomeException
ywt***@163.com
老旧小区居民楼安装电梯:
输出:
TooHandsomeException
ywt***@163.com
RUNOOB
429***[email protected]
装饰器模式允许在运行时动态地添加和移除对象的功能,而无需修改其结构。它可以避免使用子类扩展导致的类爆炸问题,使得代码更加灵活可扩展。同时,装饰器模式也遵循开闭原则,可以方便地添加新的装饰器类,而不需要修改现有的代码。
下面是一个简单的装饰器模式示例,假设有一个咖啡店,可以根据顾客的需求为咖啡添加额外的配料:
在上面的示例中,我们定义了抽象组件接口 Coffee 和具体组件类 PlainCoffee。然后,我们定义了抽象装饰器类 CoffeeDecorator,以及具体装饰器类 MilkDecorator 和 SyrupDecorator。通过不断地嵌套装饰器对象,我们可以为咖啡添加不同的配料。
RUNOOB
429***[email protected]