当前位置: 首页 > article >正文

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

目录

  • 1.概述
  • 2.类图
  • 3.应用场景及优缺点
    • 3.1 应用场景
    • 3.2 优缺点
      • 3.2.1 优点
      • 3.2.2 缺点
  • 4.实现
    • 4.1 案例类图
    • 4.2 代码实现
      • 4.2.1 定义抽象构建角色
      • 4.2.2 定义具体构建角色
      • 4.2.3 定义抽象装饰器角色
      • 4.2.4 定义具体装饰角色
      • 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接口,实现我们交通工具的基本能力—移动,这个接口就是我们装饰器模式中的抽线构建角色

public interface ITransportation {
    void move();
}

4.2.2 定义具体构建角色

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

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

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

4.2.3 定义抽象装饰器角色

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

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,在这两个具体装饰角色中添加了各自的特色功能。

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

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

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

4.2.5 装饰器模式的使用

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

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();
    }
}

运行结果如下所示:

在这里插入图片描述


http://www.kler.cn/a/380768.html

相关文章:

  • 基于python深度学习的交通标志图像识别设计与实现,卷积神经网络(CNN)作为主要架构
  • 【C++】继承的理解
  • c语言-常量和变量
  • Spring Boot框架:校园社团信息管理的现代化解决方案
  • Golang文件操作
  • 基于向量检索的RAG大模型
  • ubuntu22.04 docker-compose搭建apisix高可用
  • Spring框架的事务管理
  • 868历年真题算法设计题+程序设计题
  • leetcode-3-无重复字符的最长子串
  • Pr 视频效果:过渡
  • 使用Python Flask实战构建Web应用
  • 告别传统营销,HubSpot AI分析工具带你玩转新潮流
  • BERT预训练的MLM和NSP任务的损失函数都是什么?
  • 一文快速预览经典深度学习模型(一)——CNN、RNN、LSTM、Transformer、ViT
  • 架构师之路-学渣到学霸历程-43
  • 只允许指定ip远程连接ssh
  • 【3】流程控制
  • HarmonyOS鸿蒙开发入门,常用ArkUI组件学习(一)
  • Spring cloud
  • QT下载安装
  • 为什么要使用Docker?
  • c# 值类型
  • 青少年编程与数学 02-003 Go语言网络编程 02课题、网络分层模型
  • RHCE selinux 和 防火墙(fireword|iptable)
  • 【里程计在激光雷达SLAM中的作用】【gmapping算法hector_mapping算法】