【深入理解设计模式】装饰者设计模式

在这里插入图片描述

装饰者设计模式

装饰者设计模式(Decorator Design Pattern)是一种结构型设计模式,它允许向现有对象添加新功能而不改变其结构。这种模式通常用于需要动态地为对象添加功能或行为的情况,而且这些功能可以独立于对象本身来进行扩展。

定义:

​ 指在不改变现有对象结构的情况下,动态地给该对象增加一些职责(即增加其额外功能)的模式。

装饰(Decourator)模式中的角色:

  • Component(抽象组件角色):定义一个对象接口,可以给这些对象动态地添加职责。

  • ConcreteComponent(具体组件角色):实现Component接口,是被装饰的具体对象,可以动态地为其添加职责。

  • Decorator(抽象装饰者):维持一个指向Component对象的指针,并实现Component接口,这样可以扩展Component对象的功能。

  • ConcreteDecorator(具体装饰者):实现Decorator接口,并扩展Component对象的功能。

示例一:

快餐店有炒面、炒饭这些快餐,可以额外附加鸡蛋、火腿、培根这些配菜,当然加配菜需要额外加钱,每个配菜的价钱通常不太一样,那么计算总价就会显得比较麻烦。我们使用装饰者模式对快餐店案例进行改进,体会装饰者模式的精髓。

/*** @author OldGj 2024/02/25* @version v1.0* @apiNote 快餐类 - 抽象组件者*/
public abstract class FastFood {private float price;private String desc;public FastFood() {}public FastFood(float price, String desc) {this.price = price;this.desc = desc;}public abstract float cost();  //获取价格public float getPrice() {return price;}public void setPrice(float price) {this.price = price;}public String getDesc() {return desc;}public void setDesc(String desc) {this.desc = desc;}
}
/*** @author OldGj 2024/02/25* @version v1.0* @apiNote 炒面类 - 具体构件者(待装饰类)*/
public class FriedNoodles extends FastFood {public FriedNoodles(){super(12,"炒面");}@Overridepublic float cost() {return getPrice();}
}
/*** @author OldGj 2024/02/25* @version v1.0* @apiNote 炒饭类 - 具体构件者(待装饰类)*/
public class FriedRice extends FastFood {public FriedRice() {super(10, "炒饭");}@Overridepublic float cost() {return getPrice();}
}
/*** @author OldGj 2024/02/25* @version v1.0* @apiNote 配料类 - 抽象装饰者 继承或实现抽象组件角色并且在成员位置定义一个抽象组件角色*/
public abstract class Garnish extends FastFood {private FastFood fastFood;public Garnish(FastFood fastFood, float price, String desc) {super(price, desc);this.fastFood = fastFood;}public FastFood getFastFood() {return fastFood;}public void setFastFood(FastFood fastFood) {this.fastFood = fastFood;}
}
/*** @author OldGj 2024/02/25* @version v1.0* @apiNote 鸡蛋类 - 具体装饰者*/
public class Egg extends Garnish {@Overridepublic String getDesc() {return super.getDesc()+getFastFood().getDesc();}public Egg(FastFood fastFood) {super(fastFood, 1, "鸡蛋");}@Overridepublic float cost() {return getPrice() + getFastFood().cost();}
}
/*** @author OldGj 2024/02/25* @version v1.0* @apiNote 抽象装饰者类*/
public class Bacon extends Garnish {public Bacon(FastFood fastFood) {super(fastFood, 2, "培根");}@Overridepublic String getDesc() {return super.getDesc() + getFastFood().getDesc();}@Overridepublic float cost() {return getPrice() + getFastFood().cost();}
}
/*** @author OldGj 2024/02/25* @version v1.0* @apiNote 测试类 - 客户端*/
public class Client {public static void main(String[] args) {FastFood food = new FriedRice();System.out.println(food.getDesc() + "\t" + food.cost() + "元");System.out.println("-------------------");food = new Egg(food);System.out.println(food.getDesc() + "\t" + food.cost() + "元");System.out.println("-------------------");food = new Bacon(food);System.out.println(food.getDesc() + "\t" + food.cost() + "元");}
}

在这里插入图片描述

示例二:

考虑一个简单的例子:制作咖啡。在这个例子中,咖啡是Component,具体的咖啡类型(例如:浓缩咖啡、拿铁、摩卡)是ConcreteComponent,添加的调料(例如:牛奶、糖、巧克力)是Decorator,具体的调料种类(例如:奶泡、焦糖)是ConcreteDecorator。

让我们通过一个简单的Java示例来演示装饰者设计模式。我们将创建一个咖啡店的场景,其中有不同种类的咖啡(具体组件),以及不同种类的调料(具体装饰者)。

首先,我们定义一个接口 Coffee 作为组件,表示咖啡的基本功能:

public interface Coffee {double cost();String getDescription();
}

然后,我们实现具体的咖啡类型,例如 EspressoLatte,它们都实现了 Coffee 接口:

public class Espresso implements Coffee {@Overridepublic double cost() {return 1.99;}@Overridepublic String getDescription() {return "Espresso";}
}public class Latte implements Coffee {@Overridepublic double cost() {return 2.49;}@Overridepublic String getDescription() {return "Latte";}
}

接下来,我们创建一个装饰者 CoffeeDecorator,它也实现了 Coffee 接口,并在构造函数中接受一个 Coffee 对象,用于装饰:

public abstract class CoffeeDecorator implements Coffee {protected Coffee decoratedCoffee;public CoffeeDecorator(Coffee decoratedCoffee) {this.decoratedCoffee = decoratedCoffee;}@Overridepublic double cost() {return decoratedCoffee.cost();}@Overridepublic String getDescription() {return decoratedCoffee.getDescription();}
}

然后,我们创建具体的调料装饰者,例如 MilkCaramel

public class Milk extends CoffeeDecorator {public Milk(Coffee decoratedCoffee) {super(decoratedCoffee);}@Overridepublic double cost() {return super.cost() + 0.5; // 增加牛奶的价格}@Overridepublic String getDescription() {return super.getDescription() + ", Milk";}
}public class Caramel extends CoffeeDecorator {public Caramel(Coffee decoratedCoffee) {super(decoratedCoffee);}@Overridepublic double cost() {return super.cost() + 0.7; // 增加焦糖的价格}@Overridepublic String getDescription() {return super.getDescription() + ", Caramel";}
}

最后,我们可以在客户端代码中使用这些类来创建不同的咖啡组合:

public class Main {public static void main(String[] args) {Coffee espresso = new Espresso();System.out.println("Cost: " + espresso.cost() + ", Description: " + espresso.getDescription());Coffee latteWithMilk = new Milk(new Latte());System.out.println("Cost: " + latteWithMilk.cost() + ", Description: " + latteWithMilk.getDescription());Coffee latteWithMilkAndCaramel = new Caramel(new Milk(new Latte()));System.out.println("Cost: " + latteWithMilkAndCaramel.cost() + ", Description: " + latteWithMilkAndCaramel.getDescription());}
}

以上代码将输出:

Cost: 1.99, Description: Espresso
Cost: 2.99, Description: Latte, Milk
Cost: 3.69, Description: Latte, Milk, Caramel

这个示例演示了如何使用装饰者模式来动态地添加调料而不改变咖啡本身的结构,并且可以轻松地组合不同的调料组合。

装饰者模式是一种灵活且强大的设计模式,但它也有一些优点和缺点。

优缺点

优点:
  1. 灵活性:装饰者模式允许动态地为对象添加功能,而无需改变其原始类。这使得我们可以根据需要灵活地组合各种功能,而不会造成类爆炸。

  2. 遵循开放封闭原则:通过装饰者模式,我们可以通过添加新的装饰者来扩展功能,而无需修改现有代码,符合开放封闭原则。

  3. 单一职责原则:每个装饰者类都专注于提供单一的功能,从而遵循了单一职责原则,使得代码更易于维护和理解。

  4. 可重复使用性:由于装饰者模式通过组合的方式提供功能,因此这些装饰者可以被多次重复使用,而不会引入代码重复。

缺点:
  1. 复杂性:装饰者模式可能会导致类的数量增加,特别是在有多个装饰者时,这可能会增加代码的复杂性,使得理解和维护变得困难。

  2. 顺序依赖性:装饰者的添加顺序可能会影响最终的行为,特别是当多个装饰者相互影响时,可能需要谨慎地确定装饰者的添加顺序。

  3. 性能开销:每个装饰者都会增加额外的开销,因为它们通过组合的方式提供功能,而不是通过继承来实现。在一些性能敏感的场景中,这可能会产生一定的性能开销。

综上所述,装饰者模式在需要动态地为对象添加功能而又不想改变其原始类结构时非常有用,但需要权衡其复杂性和性能开销。

使用场景

装饰者模式通常在以下情况下被使用:

  1. 动态地为对象添加功能:当需要在运行时动态地为对象添加额外的功能,而不影响其原始类或修改其代码时,装饰者模式非常有用。例如,在不同的情况下,我们可能需要向咖啡中添加不同的调料,而不需要修改咖啡类本身。

  2. 避免类爆炸:当有许多类似但不完全相同的对象需要被创建时,使用继承可能会导致类爆炸。在这种情况下,装饰者模式可以通过组合的方式来动态地添加功能,而不是通过继承来创建大量子类。

  3. 单一职责原则:装饰者模式可以帮助我们遵循单一职责原则,即一个类应该只负责一个职责。通过将功能拆分成单独的装饰者类,每个类只负责一个功能,从而提高了代码的可维护性和可扩展性。

  4. 开放封闭原则:装饰者模式也有助于遵循开放封闭原则,即对扩展开放,对修改封闭。通过添加新的装饰者类来扩展功能,而不需要修改现有代码。

  5. 需要动态地移除功能:与添加功能类似,装饰者模式也可以用于动态地移除对象的功能。由于装饰者可以根据需要添加或移除,因此在某些情况下,这种动态性非常有用。

  6. 支持嵌套装饰:装饰者模式支持嵌套装饰,即一个装饰者可以装饰另一个装饰者,从而形成复杂的功能组合。

总的来说,装饰者模式适用于需要动态地添加或移除对象功能,并且希望避免类爆炸以及遵循单一职责和开放封闭原则的情况下。它提供了一种灵活且可扩展的方式来组合对象功能,而不需要修改其原始类结构。

JDK源码解析

IO流中的包装类使用到了装饰者模式。BufferedInputStream,BufferedOutputStream,BufferedReader,BufferedWriter。

我们以BufferedWriter举例来说明,先看看如何使用BufferedWriter

public class Demo {public static void main(String[] args) throws Exception{//创建BufferedWriter对象//创建FileWriter对象FileWriter fw = new FileWriter("C:\\Users\\Think\\Desktop\\a.txt");BufferedWriter bw = new BufferedWriter(fw);//写数据bw.write("hello Buffered");bw.close();}
}

使用起来感觉确实像是装饰者模式,接下来看它们的结构:

小结:

​ BufferedWriter使用装饰者模式对Writer子实现类进行了增强,添加了缓冲区,提高了写数据的效率。

装饰者模式(Decorator Pattern)和代理模式(Proxy Pattern)的区别:

装饰者模式和代理模式是两种常见的设计模式,它们在实现上有一些区别:

装饰者模式:

  1. 用途:装饰者模式用于动态地为对象添加额外的功能,而不修改其原始类的结构。它通过组合的方式实现功能的添加和组合。

  2. 结构:装饰者模式通常包括一个抽象组件(Component)、具体组件(Concrete Component)、装饰者(Decorator)和具体装饰者(Concrete Decorator)。装饰者维护一个指向组件对象的引用,并实现与组件相同的接口,通过递归调用实现功能的装饰。

  3. 动态性:装饰者模式允许在运行时动态地添加或移除功能,因为装饰者对象可以随时添加到对象上。

代理模式:

  1. 用途:代理模式用于控制对对象的访问。代理通常在客户端和实际对象之间起到中介的作用,可以用于实现延迟加载、访问控制、远程代理等场景。

  2. 结构:代理模式通常包括一个抽象主题(Subject)、具体主题(Real Subject)和代理(Proxy)。代理实现与主题相同的接口,并在内部持有一个真实主题的引用,在方法调用时代理可以在调用前后进行额外的处理。

  3. 目的:代理模式的主要目的是控制对对象的访问,而不是为对象添加功能。代理可以在客户端和实际对象之间提供间接层,从而实现对实际对象的控制。

总结:

  • 装饰者模式用于动态地为对象添加功能,而代理模式用于控制对对象的访问。
  • 装饰者模式通过组合的方式实现功能的添加和组合,而代理模式则通常通过在客户端和实际对象之间添加一个代理对象来实现控制。
  • 装饰者模式主要关注功能的增强和组合,而代理模式主要关注对对象的访问控制。
  • 获取目标对象构建的地方不同
    装饰者是由外界传递进来,可以通过构造方法传递
    静态代理是在代理类内部创建,以此来隐藏目标对象

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

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

相关文章

【计算机网络】1 因特网概述

一.网络、互联网和因特网 1.网络(network),由若干结点(node)和连接这些结点的链路(link)组成。 2.多个网络还可以通过路由器互联起来,这样就构成了一个覆盖范围更大的网络&#xf…

【卡码网】完全背包问题 52. 携带研究材料——代码随想录算法训练营Day44

题目链接:题目页面 题目描述 题目描述 小明是一位科学家,他需要参加一场重要的国际科学大会,以展示自己的最新研究成果。他需要带一些研究材料,但是他的行李箱空间有限。这些研究材料包括实验设备、文献资料和实验样本等等&…

终于明白kmp算法

在刷代码随想录的时候,遇到了leetcode这道经典题目 28. 实现 strStr() 力扣题目链接(opens new window) 实现 strStr() 函数。 给定一个 haystack 字符串和一个 needle 字符串,在 haystack 字符串中找出 needle 字符串出现的第一个位置 (从0开始)。如…

Selenium IDE插件录制网页,解放双手

1、 国内下载地址 https://www.crx4chrome.com/crx/77585/ ,这个网络正常基本可以下载,目前最新版本是3.17.2。 点击Crx4Chrome下载。下载后的文件名称是:mooikfkahbdckldjjndioackbalphokd-3.17.2-Crx4Chrome.com.crx。 2、 安装 直接打开…

webpack 如何实现模块懒加载

首先在打包的时候使用 splitwebpackplugin 进行分割 在实际引用中,webpack 实现模块的懒加载是通过动态导入(dynamic import )来实现的。动态导入是 es6 的一项功能,允许在运行时异步加载模块,从而实现按需加载。 使…

CMS垃圾回收器

CMS垃圾回收 CMS GC的官方名称为“Mostly Concurrenct Mark and Sweep Garbage Collector”(最大-并发-标记-清除-垃圾收集器)。 作用范围: 老年代 算法: 并发标记清除算法。 启用参数:-XX:UseConMarkSweepGC 默认回收…

探索创造无限可能——Autodesk AutoCAD 2022(CAD 2022)系统要求

随着科技的不断进步和发展,计算机辅助设计(CAD)已经成为现代设计行业中不可或缺的一部分。在众多CAD软件中,Autodesk AutoCAD 2022(CAD 2022)无疑是最受欢迎和广泛应用的一款软件。作为一款全球领先的CAD软…

打造多平台游戏,Pygame让梦想照进现实

Pygame是一个流行的Python库,用于开发2D游戏。尽管它主要用于桌面游戏,但通过一些额外的工具和技巧,你也可以使用Pygame来打造多平台游戏,比如支持Windows、Linux、macOS、Android和iOS等平台。 下面是一个简单的Pygame游戏示例&…

sql 行列互换

在SQL中进行行列互换可以使用PIVOT函数。下面是一个示例查询及其对应的结果: 创建测试表格 CREATE TABLE test_table (id INT PRIMARY KEY,name VARCHAR(50),category VARCHAR(50) );向测试表格插入数据 INSERT INTO test_table VALUES (1, A, Category A); INSE…

某电力铁塔安全监测预警系统案例分享

项目概述 电力铁塔是承载电力供应的重要设施,它的安全性需要得到可靠的保障。但是铁塔一般安装在户外,分布广泛,且有很多安装在偏远地区,容易受到自然、人力的影响和破环。因此需要使用辅助的方法实时监控铁塔的安全状态&#xff…

ABC342 题解

ABC342 题解 A Description 给定一个串 s s s,求与 s s s 中其它字符不同的唯一字符的编号。 Solution 直接把 s s s 存到 t t t 里排序(确实,sort(t.begin(), t.end()) 就可以),首尾特判再到 s s s 里查一下…

计算机设计大赛 深度学习大数据物流平台 python

文章目录 0 前言1 课题背景2 物流大数据平台的架构与设计3 智能车货匹配推荐算法的实现**1\. 问题陈述****2\. 算法模型**3\. 模型构建总览 **4 司机标签体系的搭建及算法****1\. 冷启动**2\. LSTM多标签模型算法 5 货运价格预测6 总结7 部分核心代码8 最后 0 前言 &#x1f5…

office word保存pdf高质量设置

1 采用第三方pdf功能生成 分辨率越大质量越好

MySQL集群 双主架构(配置命令)

CSDN 成就一亿技术人&#xff01; 今天刚开学第一天给大家分享一期&#xff1a;MySQL集群双主的配置需求和命令 CSDN 成就一亿技术人&#xff01; 神秘泣男子主页&#xff1a;作者首页 <———— MySQL专栏 &#xff1a;MySQL数据库专栏<———— MySQL双主是一…

学习笔记丨Shell

Usage of shell script References: Learn Shell, Shell 教程 | 菜鸟教程 The first line of shell script file begins with #!, followed by the full path where the shell interpreter is located. For example, #!/bin/bashTo find out the currently active shell and i…

python语言常见面试题:描述Python中的字典(Dictionary)和集合(Set)之间的区别。

Python中的字典&#xff08;Dictionary&#xff09;和集合&#xff08;Set&#xff09;是两种非常有用的数据结构&#xff0c;它们之间有一些明显的区别。 字典&#xff08;Dictionary&#xff09; 字典是一种无序的键值对集合。在字典中&#xff0c;每个键&#xff08;key&a…

Oracle 基础表管理(Heap-Organized Table Management)

表是数据库中负责数据存储的对象&#xff0c;在RDBMS中&#xff0c;数据以行、列的形式存储在表中。Oracle中表有很多种类型&#xff0c;最基础且应用最常用的类型就是堆表&#xff08;Heap-Organized Table&#xff09;&#xff0c;本文列举了Oracle堆表的常用管理操作。 一、…

Cpython和Jpython区别

Cpython和Jpython是Python语言的两种不同实现方式&#xff0c;它们之间存在一些关键的区别。 实现语言&#xff1a;Cpython是用C语言实现的&#xff0c;而Jpython则是用Java语言实现的。这意味着Cpython的源代码是用C语言编写的&#xff0c;而Jpython的源代码是用Java语言编写的…

pytorch --反向传播和优化器

1. 反向传播 计算当前张量的梯度 Tensor.backward(gradientNone, retain_graphNone, create_graphFalse, inputsNone)计算当前张量相对于图中叶子节点的梯度。 使用反向传播&#xff0c;每个节点的梯度&#xff0c;根据梯度进行参数优化&#xff0c;最后使得损失最小化 代码…

React Hooks概述及常用的React Hooks介绍

Hook可以让你在不编写class的情况下使用state以及其他React特性 useState ● useState就是一个Hook ● 通过在函数组件里调用它来给组件添加一些内部state,React会在重复渲染时保留这个state 纯函数组件没有状态&#xff0c;useState()用于设置和使用组件的状态属性。语法如下…