Java设计模式:深入装饰器模式的三种写法(六)

在这里插入图片描述

码到三十五 : 个人主页

心中有诗画,指尖舞代码,目光览世界,步履越千山,人间尽值得 !


在软件设计中,装饰器模式是一种结构型设计模式,它允许用户在不改变对象结构的情况下,动态地给对象添加一些额外的职责。在Java中,装饰器模式通常通过实现共同的接口或使用继承来扩展对象的行为。

[参见]:

Java设计模式:核心概述(一)

Java设计模式:单例模式之六种实现方式详解(二)

Java设计模式:工厂模式之简单工厂、工厂方法、抽象工厂(三)

Java设计模式:建造者模式之经典与流式的三种实现(四)

Java设计模式:适配器模式的三种形式(五)

目录

    • 前言
    • 一、装饰器模式的主要组成
    • 二、装饰器模式的优点
    • 三、装饰器模式的局限
    • 四、装饰器模式的使用场景
    • 五、装饰器模式的三种写法
      • 方式1️⃣:使用接口和抽象类【标准实现】
        • 步骤 1:定义抽象组件
        • 步骤 2:创建具体组件
        • 步骤 3:定义抽象装饰器
        • 步骤 4:创建具体装饰器
        • 步骤 5:客户端代码
      • 方式2️⃣:使用内部类
      • 方式3️⃣:使用Java 8的函数式接口和Lambda表达式
    • 总结

前言

Java中的装饰器模式是一种结构型设计模式,它允许你在不修改现有类的情况下,动态地将新功能添加到对象上。装饰器模式通过创建一个包装了原始对象的装饰器类来实现这一点。装饰器类与原始类具有相同的接口,因此它们可以互换使用。

一、装饰器模式的主要组成

  1. 抽象组件(Component):定义了一个接口,用于规定具体组件和装饰器类的共同行为。

  2. 具体组件(ConcreteComponent):实现了抽象组件接口,它是被装饰的原始对象,通常可以独立使用。

  3. 抽象装饰器(Decorator):这是一个抽象类,它实现了抽象组件接口,并包含一个对抽象组件的引用。抽象装饰器通常提供对所有方法的默认实现,这些方法通常只是简单地调用被装饰对象(即抽象组件引用)上的相应方法。

  4. 具体装饰器(ConcreteDecorator):这是抽象装饰器的子类,它负责添加新的功能。具体装饰器可以重写父类(抽象装饰器)的方法,以在被装饰对象的方法调用前后增加额外的行为。

二、装饰器模式的优点

  • 灵活性:装饰器模式提供了比继承更灵活的扩展方式。你可以通过组合多个装饰器来创建具有各种功能的对象,而无需修改原始类或使用大量的子类。
  • 避免类爆炸:当使用继承来添加功能时,每个新功能都可能需要一个新的子类。这可能导致类数量的快速增长,使系统变得复杂且难以维护。装饰器模式通过动态地添加功能来避免这个问题。
  • 开闭原则:装饰器模式符合开闭原则,即对扩展开放,对修改关闭。你可以添加新的装饰器来扩展功能,而无需修改现有的代码。

三、装饰器模式的局限

  • 额外的复杂性:使用装饰器模式可能会增加系统的复杂性,因为你需要管理额外的装饰器类和对象。此外,理解装饰器之间的交互和它们如何影响被装饰对象的行为可能需要一些努力。
  • 类型匹配问题:在某些情况下,装饰器可能会破坏类型匹配。例如,如果你有一个需要特定类型参数的方法,并且你传递了一个被装饰的对象(其类型是装饰器类型),那么可能会出现类型不匹配的问题。这可以通过使用接口和泛型来缓解。
  • 额外的性能开销:由于装饰器可能会创建多层对象包装,因此在某些情况下可能会引入额外的性能开销。

四、装饰器模式的使用场景

  • 需要动态地添加或撤销功能:当你需要在运行时根据需要动态地添加或撤销功能时,装饰器模式是一个很好的选择。例如,你可以创建一个装饰器来记录方法调用的日志,然后在需要时将其应用到对象上。
  • 避免使用大量的子类:当你想要扩展一个类的功能,但又不希望创建大量的子类时,可以使用装饰器模式。通过创建装饰器类来添加新功能,你可以避免类数量的快速增长。
  • 需要透明的扩展功能:装饰器模式允许你在不修改原始类的情况下透明地扩展功能。这意味着你可以在不改变客户端代码的情况下添加新功能。客户端代码可以继续使用原始类的接口,而无需了解装饰器的存在。

五、装饰器模式的三种写法

方式1️⃣:使用接口和抽象类【标准实现】

下面是一个简单的Java装饰器模式的实现:
在这里插入图片描述

步骤 1:定义抽象组件
public interface Coffee {double getCost();String getIngredients();
}
步骤 2:创建具体组件
public class SimpleCoffee implements Coffee {@Overridepublic double getCost() {return 1;}@Overridepublic String getIngredients() {return "Coffee";}
}
步骤 3:定义抽象装饰器
public abstract class CoffeeDecorator implements Coffee {protected final Coffee decoratedCoffee;public CoffeeDecorator(Coffee coffeeToDecorate) {this.decoratedCoffee = coffeeToDecorate;}@Overridepublic double getCost() {return decoratedCoffee.getCost();}@Overridepublic String getIngredients() {return decoratedCoffee.getIngredients();}
}
步骤 4:创建具体装饰器
public class MilkCoffee extends CoffeeDecorator {public MilkCoffee(Coffee coffeeToDecorate) {super(coffeeToDecorate);}@Overridepublic double getCost() {return decoratedCoffee.getCost() + 0.5;}@Overridepublic String getIngredients() {return decoratedCoffee.getIngredients() + ", Milk";}
}public class WhipCoffee extends CoffeeDecorator {public WhipCoffee(Coffee coffeeToDecorate) {super(coffeeToDecorate);}@Overridepublic double getCost() {return decoratedCoffee.getCost() + 0.7;}@Overridepublic String getIngredients() {return decoratedCoffee.getIngredients() + ", Whip";}
}
步骤 5:客户端代码
public class CoffeeShop {public static void main(String[] args) {Coffee simpleCoffee = new SimpleCoffee();System.out.println("Cost: " + simpleCoffee.getCost() + "; Ingredients: " + simpleCoffee.getIngredients());Coffee milkCoffee = new MilkCoffee(simpleCoffee);System.out.println("Cost: " + milkCoffee.getCost() + "; Ingredients: " + milkCoffee.getIngredients());Coffee milkWhipCoffee = new WhipCoffee(milkCoffee);System.out.println("Cost: " + milkWhipCoffee.getCost() + "; Ingredients: " + milkWhipCoffee.getIngredients());}
}

运行上面的客户端代码,你会看到不同的咖啡组合和它们的价格以及配料。

方式2️⃣:使用内部类

有时候,装饰器可能只针对某个特定的组件类有用。在这种情况下,可以将装饰器类作为组件类的内部类来实现。

public class Component {public void operation() {// ...}public class Decorator extends Component {@Overridepublic void operation() {super.operation();// Additional functionality}}
}

然后客户端可以这样使用:

Component component = new Component();
Component decoratedComponent = component.new Decorator();
decoratedComponent.operation();

这种方式不太常见,因为它将装饰器和被装饰的组件紧密耦合在一起。然而,在某些情况下,如果装饰器的逻辑与被装饰的组件紧密相关,并且不打算与其他组件共享,这种方式可能是合适的。

方式3️⃣:使用Java 8的函数式接口和Lambda表达式

在Java 8及更高版本中,可以利用函数式接口和Lambda表达式来更简洁地实现装饰器模式。例如,可以定义一个函数式接口来表示组件的操作:

@FunctionalInterface
public interface ComponentOperation {void execute();
}

然后创建一个具体的组件实现:

public class ConcreteComponent implements ComponentOperation {@Overridepublic void execute() {System.out.println("Executing operation in ConcreteComponent");}
}

现在,可以创建一个装饰器类,它接受一个ComponentOperation并添加额外的行为:

public class Decorator implements ComponentOperation {private final ComponentOperation componentOperation;private final Runnable additionalBehavior;public Decorator(ComponentOperation componentOperation, Runnable additionalBehavior) {this.componentOperation = componentOperation;this.additionalBehavior = additionalBehavior;}@Overridepublic void execute() {additionalBehavior.run(); // This could be before or after the component operationcomponentOperation.execute();// additionalBehavior.run(); // Optionally call additional behavior after the operation}
}

客户端代码可以像这样使用装饰器:

public class Client {public static void main(String[] args) {ComponentOperation component = new ConcreteComponent();Runnable additionalBehavior = () -> System.out.println("Executing additional behavior");ComponentOperation decoratedComponent = new Decorator(component, additionalBehavior);decoratedComponent.execute();}
}

在这个例子中,我们没有使用继承或抽象类,而是使用了组合和Java 8的函数式接口来实现装饰器模式。这种方式更加灵活,并且允许在运行时动态地添加不同的行为。

总结

装饰器模式是一种强大的设计工具,它允许开发人员在不修改现有类的情况下动态地扩展对象的行为。通过合理地使用装饰器模式,可以构建出更加灵活、可扩展和可维护的软件系统。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/526911.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

如何使用固定公网地址SFTP远程传输文件至安卓Termux本地目录?

文章目录 1. 安装openSSH2. 安装cpolar3. 远程SFTP连接配置4. 远程SFTP访问4. 配置固定远程连接地址 SFTP(SSH File Transfer Protocol)是一种基于SSH(Secure Shell)安全协议的文件传输协议。与FTP协议相比,SFTP使用了…

0103n阶行列式-行列式-线性代数

文章目录 一 n阶行列式二 三阶行列式三 特殊行列式结语 一 n阶行列式 ∣ a 11 a 12 ⋯ a 1 n a 21 a 22 ⋯ a 2 n ⋯ ⋯ ⋯ ⋯ a n 1 a n 2 ⋯ a n n ∣ \begin{vmatrix}a_{11}&a_{12}&\cdots&a_{1n}\\a_{21}&a_{22}&\cdots&a_{2n}\\\cdots&\cdots…

【硬件基础】H桥驱动芯片举例

1、TB6612电机驱动芯片 TB6612芯片是一款双路H桥型的直流电机驱动芯片,可以驱动两个直流电机并控制其转速和方向。 参考:C141517_电机驱动芯片_TB6612FNG,C,8,EL_规格书_TOSHIBA(东芝)电机驱动芯片规格书 1.1如下是芯片的引脚图: VM&#…

js【详解】async await

为什么要使用 async await async await 实现了使用同步的语法实现异步,不再需要借助回调函数,让代码更加易于理解和维护。 (async function () {// await 必须放在 async 函数中try {// 加载第一张图片const img1 await loadImg1()// 加载第二张图片co…

【ghost】制作一个DOS启动盘用于备份/恢复系统

常用的DOS启动盘制作工具有USBoot、Ghost及FlashBoot等,本次DOS启动盘使用Ghost工具制作。 制作前准备 装有win10(或win7)系统的PC机,1台;U盘,1个;(建议用户选择兼容性较高的金士顿U盘;此次演…

探索云原生数据库技术:构建高效可靠的云原生应用

数据库是应用开发中非常重要的组成部分,可以进行数据的存储和管理。随着企业业务向数字化、在线化和智能化的演进过程中,面对指数级递增的海量存储需求和挑战以及业务带来的更多的热点事件、突发流量的挑战,传统的数据库已经很难满足和响应快…

从一个励志故事发现了特别牛掰的搭建静态网站的开源工具Docusaurus,很上头

起初只是看到这篇CSDN推送的励志故事,突然发现Docusaurus很合我意,但在简中范围内查询了一下发现东西不少,但都只聊得比较肤浅,不能让小白很快理解,不过建议还是应该看一下Docusaurus术语相关的分享,下面是…

数据库 — 增删查改

一、操作数据库、表 显示 show databases;创建 create database xxx;使用 use xxx; 删除 drop database xxx;查看表; show tables; 查看表结构 desc 表名; 创建 create table 表名(字段1 类型1,字段2 类型2,.... ); 删除 drop table 表名; 二…

维修家用美容射频美容仪

今天收到客户寄过来的一款家用射频美容仪。根据客户的反馈,插电开机没反应,经过排查,原来是12v-2A电源坏了。给客户更换一个新电源就可以了。

Swift 入门学习:集合(Collection)类型趣谈-下

概览 集合的概念在任何编程语言中都占有重要的位置,正所谓:“古来聚散地,宿昔长荆棘;游人聚散中,一片湖光里”。把那一片片、一瓣瓣、一粒粒“可耐”的小精灵全部收拢、吸纳的井然有序、条条有理,怎能不让…

数据库的联表查询

多表查询和子查询 多表查询和子查询是解决复杂查询问题的两种常用方法。 【1】子查询 就相当于是我们在一步步解决问题 将一条SQL语句的查询结果括号当做另一条SQL语句的查询条件 -- 子查询select * form * where *;select * from * where (select * from * where *;); ​…

20240310-1-Java后端开发知识体系

Java 基础 知识体系 Questions 1. HashMap 1.8与1.7的区别 1.71.8底层结构数组链表数组链表/红黑树插入方式头插法尾插法计算hash值4次位运算5次异或运算1次位运算1次异或运算扩容、插入先扩容再插入先插入再扩容扩容后位置计算重新hash原位置或原位置旧容量 (1) 扩容因子…