掌握聚合最新动态了解行业最新趋势
API接口,开发服务,免费咨询服务

设计模式之装饰器模式详解(定义、优缺点、应用场景、实例类图)

装饰器模式(Decorator Pattern)允许向一个现有的对象添加新的功能,同时又不改变其结构。这种类型的设计模式属于结构型模式,它是作为现有的类的一个包装。装饰器模式通过将对象包装在装饰器类中,以便动态地修改其行为。这种模式创建了一个装饰类,用来包装原有的类,并在保持类方法签名完整性的前提下,提供了额外的功能。

我们通过下面的实例来演示装饰器模式的用法。其中,我们将把一个形状装饰上不同的颜色,同时又不改变形状类。

一、装饰器模式的概要

意图

动态地给一个对象添加额外的职责,同时不改变其结构。装饰器模式提供了一种灵活的替代继承方式来扩展功能。

主要解决的问题

  1. 避免通过继承引入静态特征,特别是在子类数量急剧膨胀的情况下。

  2. 允许在运行时动态地添加或修改对象的功能。

结构

装饰器模式包含以下几个核心角色:

  1. 抽象组件(Component):定义了原始对象和装饰器对象的公共接口或抽象类,可以是具体组件类的父类或接口。

  2. 具体组件(Concrete Component):是被装饰的原始对象,它定义了需要添加新功能的对象。

  3. 抽象装饰器(Decorator):继承自抽象组件,它包含了一个抽象组件对象,并定义了与抽象组件相同的接口,同时可以通过组合方式持有其他装饰器对象。

  4. 具体装饰器(Concrete Decorator):实现了抽象装饰器的接口,负责向抽象组件添加新的功能。具体装饰器通常会在调用原始对象的方法之前或之后执行自己的操作。

装饰器模式通过嵌套包装多个装饰器对象,可以实现多层次的功能增强。每个具体装饰器类都可以选择性地增加新的功能,同时保持对象接口的一致性。

使用场景

  1. 当需要在不增加大量子类的情况下扩展类的功能。

  2. 当需要动态地添加或撤销对象的功能。

实现方式

  1. 定义组件接口:创建一个接口,规定可以动态添加职责的对象的标准。

  2. 创建具体组件:实现该接口的具体类,提供基本功能。

  3. 创建抽象装饰者:实现同样的接口,持有一个组件接口的引用,可以在任何时候动态地添加功能。

  4. 创建具体装饰者:扩展抽象装饰者,添加额外的职责。

关键代码

  1. Component接口:定义了可以被装饰的对象的标准。

  2. ConcreteComponent类:实现Component接口的具体类。

  3. Decorator抽象类:实现Component接口,并包含一个Component接口的引用。

  4. ConcreteDecorator类:扩展Decorator类,添加额外的功能。

应用实例

  1. 孙悟空的72变:孙悟空(ConcreteComponent)通过变化(Decorator)获得新的能力。

  2. 画框装饰画:一幅画(ConcreteComponent)可以通过添加玻璃(ConcreteDecorator)和画框(ConcreteDecorator)来增强其展示效果。

二、装饰器模式的优缺点

优点

  1. 低耦合:装饰类和被装饰类可以独立变化,互不影响。

  2. 灵活性:可以动态地添加或撤销功能。

  3. 替代继承:提供了一种继承之外的扩展对象功能的方式。

缺点

  1. 复杂性:多层装饰可能导致系统复杂性增加。

三、装饰器模式的使用建议

  1. 在需要动态扩展功能时,考虑使用装饰器模式。

  2. 保持装饰者和具体组件的接口一致,以确保灵活性。

  3. 装饰器模式可以替代继承,但应谨慎使用,避免过度装饰导致系统复杂。

四、装饰器模式的实现

我们将创建一个 Shape 接口和实现了 Shape 接口的实体类。然后我们创建一个实现了 Shape 接口的抽象装饰类 ShapeDecorator,并把 Shape 对象作为它的实例变量。RedShapeDecorator 是实现了 ShapeDecorator 的实体类。

DecoratorPatternDemo 类使用 RedShapeDecorator 来装饰 Shape 对象。

装饰器模式

  1. 步骤 1:创建一个接口:

Shape.java

public interface Shape {
   void draw();
}
  1. 步骤 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");
   }
}
  1. 步骤 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");
   }
}
  1. 步骤 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();
   }
}
  1. 步骤 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

声明:所有来源为“聚合数据”的内容信息,未经本网许可,不得转载!如对内容有异议或投诉,请与我们联系。邮箱:marketing@think-land.com

  • 全球天气预报

    支持全球约2.4万个城市地区天气查询,如:天气实况、逐日天气预报、24小时历史天气等

    支持全球约2.4万个城市地区天气查询,如:天气实况、逐日天气预报、24小时历史天气等

  • 购物小票识别

    支持识别各类商场、超市及药店的购物小票,包括店名、单号、总金额、消费时间、明细商品名称、单价、数量、金额等信息,可用于商品售卖信息统计、购物中心用户积分兑换及企业内部报销等场景

    支持识别各类商场、超市及药店的购物小票,包括店名、单号、总金额、消费时间、明细商品名称、单价、数量、金额等信息,可用于商品售卖信息统计、购物中心用户积分兑换及企业内部报销等场景

  • 涉农贷款地址识别

    涉农贷款地址识别,支持对私和对公两种方式。输入地址的行政区划越完整,识别准确度越高。

    涉农贷款地址识别,支持对私和对公两种方式。输入地址的行政区划越完整,识别准确度越高。

  • 人脸四要素

    根据给定的手机号、姓名、身份证、人像图片核验是否一致

    根据给定的手机号、姓名、身份证、人像图片核验是否一致

  • 个人/企业涉诉查询

    通过企业关键词查询企业涉讼详情,如裁判文书、开庭公告、执行公告、失信公告、案件流程等等。

    通过企业关键词查询企业涉讼详情,如裁判文书、开庭公告、执行公告、失信公告、案件流程等等。

0512-88869195
数 据 驱 动 未 来
Data Drives The Future