设计模式之结构型模式---装饰器模式

目录

  • 1.概述
  • 2.类图
  • 3.应用场景及优缺点
    • [3.1 应用场景](#3.1 应用场景)
    • [3.2 优缺点](#3.2 优缺点)
      • [3.2.1 优点](#3.2.1 优点)
      • [3.2.2 缺点](#3.2.2 缺点)
  • 4.实现
    • [4.1 案例类图](#4.1 案例类图)
    • [4.2 代码实现](#4.2 代码实现)
      • [4.2.1 定义抽象构建角色](#4.2.1 定义抽象构建角色)
      • [4.2.2 定义具体构建角色](#4.2.2 定义具体构建角色)
      • [4.2.3 定义抽象装饰器角色](#4.2.3 定义抽象装饰器角色)
      • [4.2.4 定义具体装饰角色](#4.2.4 定义具体装饰角色)
      • [4.2.5 装饰器模式的使用](#4.2.5 装饰器模式的使用)

1.概述

装饰器模式是指在不改变现有对象结构的情况下,动态的给对象增加一些职责。它是一种用于替代继承的技术,通过一种无需定义子类的方式给对象动态增加职责,使用对象之间的关联关系取代类之间的继承关系。装饰器模式通过引入一个装饰类,在装饰类中既可以调用待装饰的原有类的方法,还可以增加新的方法,达到扩展原有功能的目的。例如造一个交通工具,这种交通工具刚开始时只能是在陆地上跑,也就是咱们的汽车,但是随着技术的发展,我们可以给我们的汽车增加在水中移动的功能,或者是在天上飞的功能。这就是装饰器模式的场景之一。后面会用Java语言的代码来实现这个例子。

2.类图

装饰器模式的主要角色有如下几个:

抽象构建角色(Component): 定义一个抽象接口,规范准备接收附加责任的对象
具体构建角色(ConcreteComponent): 实现抽象构建角色定义的接口。通过装饰器角色为其添加职责
抽象装饰器角色(Decorator) 实现抽象构建中定义的接口,并包含具体构建角色的实例,可以通过其子类扩展具体构建角色实例的功能
具体装饰角色(ConcreteDecorator1、ConcreteDecorator2): 实现抽象装饰的相关方法,并给具体构建角色添加新的职责(功能)

3.应用场景及优缺点

3.1 应用场景

当不能采用继承的方式扩展功能或者是采用继承的方式扩展功能时不好维护的情况下就可以使用装饰器模式。比如一个功能的定义时就有基础版,加强版,VIP版本等基于基础功能扩展出新功能的情况下,我们就可以采用装饰器模式。因为装饰器模式完全遵守开闭原则,即对修改关闭,对扩展开放。

3.2 优缺点

3.2.1 优点

装饰器模式的优点有四个,如下:

  1. 对于扩展对象的功能,装饰器模式比继承更加灵活,因为装饰器模式不会导致类的个数急剧增加。
  2. 使用装饰器模式可以让我们通过一种动态的方式扩展一个对象的功能,比如我们可以通过配置文件在运行的时候选择不同的具体装饰类,从而实现不同的行为
  3. 装饰器可以对一个对象进行多次装饰
  4. 具体构建类与具体装饰类可以独立变化,用户可以更具需要增加的具体构建类和具体装饰类。符合开闭原则

3.2.2 缺点

装饰器模式的缺点主要有两个,如下:

  1. 使用装饰器模式进行系统设计时将会产生很多小对象,大量的小对象会占用更多的系统资源,在一定程度上会影响程序的性能
  2. 装饰器模式使用起来比继承更容易出错,排除错误也会更困难,特别是对于多次装饰的对象,调试时寻找错误可能需要逐级排查。

4.实现

我们以一个汽车改造成水陆两用汽车,陆空两用汽车的例子介绍装饰器模式在程序中的实现,我们用Java语言实现。

假设我们要造一个代步的交通工具,这个交通工具第一版本只需要在陆地上跑就行,也就是我们的汽车,然后随着技术增长,我们需要为我们的汽车扩展在水中行驶的能力和在空中飞行的能力,使用Java代码实现如下所示:

4.1 案例类图

4.2 代码实现

4.2.1 定义抽象构建角色

首先我们定义一个ITransportation接口,实现我们交通工具的基本能力---移动,这个接口就是我们装饰器模式中的抽线构建角色

java 复制代码
public interface ITransportation {
    void move();
}

4.2.2 定义具体构建角色

定义好抽象构建角色后,我们就可以定义具体构建角色,也就是我们的汽车类了,在汽车类里实现了抽象构建角色定义的功能

java 复制代码
public class Car implements ITransportation {
    public Car(){
        System.out.println("我只是单纯的一辆车");
    }

    @Override
    public void move() {
        System.out.println("我可以在陆地上移动");
    }
}

4.2.3 定义抽象装饰器角色

接下来就需要定义抽象装饰器角色准备扩展功能,为了能扩展汽车的能力,我们提供了一个Transformer类,这个类也实现了抽象构建角色中的ITransportation接口,并且将抽象构建角色对应的具体构建角色通过构造函数注入到Transformer类中,这里的Transformer类就相当于抽象装饰器角色。我们需要的扩展功能后的对象都需要继承它。

java 复制代码
public class Transformer implements ITransportation {
    private final ITransportation mTransportation;

    public Transformer(ITransportation transportation){
        this.mTransportation = transportation;
    }

    @Override
    public void move() {
        mTransportation.move();
    }
}

4.2.4 定义具体装饰角色

具体装饰角色有两个,分别是水陆两用的汽车AmphibiousCar ,陆空两用的汽车FlyingCar,在这两个具体装饰角色中添加了各自的特色功能。

java 复制代码
public class AmphibiousCar extends Transformer{
    public AmphibiousCar(ITransportation transportation) {
        super(transportation);
        System.out.println("我是水陆两用汽车");
    }

    public void moveInWater(){
        System.out.println("我可以在水里跑");
    }
}
java 复制代码
public class FlyingCar extends Transformer{
    public FlyingCar(ITransportation transportation) {
        super(transportation);
        System.out.println("我是陆空两用汽车");
    }

    public void flying(){
        System.out.println("我可以在天上飞");
    }
}

4.2.5 装饰器模式的使用

使用时我们先通过抽象构建角色创建出具体构建角色的对象,这个对象时需要被装饰的对象,然后将其传递到陆空两用车类的构造函数中和水陆两用车类的构造函数中,分别创建出陆空两用车的对象,水陆两用车的对象。然后调用对应的方法实现各自的能力

java 复制代码
public class Client {
    public static void main(String[] args) {
        ITransportation myCar = new Car();
        myCar.move();
        System.out.println("============================");

        System.out.println("===========陆空两用车=============");
        FlyingCar flyingCar = new FlyingCar(myCar);
        flyingCar.move();
        flyingCar.flying();

        System.out.println("=============水陆两用车======================");
        AmphibiousCar amphibiousCar = new AmphibiousCar(myCar);
        amphibiousCar.move();
        amphibiousCar.moveInWater();
    }
}

运行结果如下所示:

相关推荐
尽兴-1 分钟前
Redis模拟延时队列 实现日程提醒
java·redis·java-rocketmq·mq
书埋不住我22 分钟前
java第三章
java·开发语言·servlet
boy快快长大24 分钟前
将大模型生成数据存入Excel,并用增量的方式存入Excel
java·数据库·excel
孟秋与你27 分钟前
【spring】spring单例模式与锁对象作用域的分析
java·spring·单例模式
菜菜-plus30 分钟前
java 设计模式 模板方法模式
java·设计模式·模板方法模式
萨达大31 分钟前
23种设计模式-模板方法(Template Method)设计模式
java·c++·设计模式·软考·模板方法模式·软件设计师·行为型设计模式
tian-ming33 分钟前
(十八)JavaWeb后端开发案例——会话/yml/过滤器/拦截器
java·开发语言·前端
不能只会打代码36 分钟前
大学课程项目中的记忆深刻 Bug —— 一次意外的数组越界
java·github·intellij-idea·话题博客
深海呐39 分钟前
Android 从本地选择视频,用APP播放或进行其他处理
android·音视频·从本地选择视频,用app播放·从本地选择视频,并拿到信息·跳转到本地视频列表
深海呐39 分钟前
Android Google登录接入
android·google登录接入·android 谷歌登录接入·google登录·android google