设计模式篇(Java):装饰者模式

news2024/11/25 3:06:47

👨‍💻本文专栏:设计模式篇-装饰者模式
👨‍💻本文简述:装饰者模式的详解以及jdk中的应用
👨‍💻上一篇文章: 设计模式篇(Java):桥接模式
👨‍💻有任何问题,都可以私聊我,文章最后有vx名片。感谢支持!
🦹知道的越多,不知道的越多!!!不能停下学习的脚步

在这里插入图片描述

十、装饰者模式

10.1 装饰者模式基本介绍

引出装饰者模式的示例

咖啡吧订单问题:

  • 咖啡种类/单品咖啡:Espresso(意大利浓咖啡)、ShortBlack、LongBlack(美式 咖啡)、Decaf(无因咖啡)
  • 调料:Milk、Soy(豆浆)、Chocolate
  • 要求在扩展新的咖啡种类时,具有良好的扩展性、改动方便、维护方便
  • 使用OO的来计算不同种类咖啡的费用: 客户可以点单品咖啡,也可以单品咖啡+调料组合

那么使用传统方法(最笨的)

在这里插入图片描述

问题分析

  • Drink 是一个抽象类,表示饮料
  • des就是对咖啡的描述, 比如咖啡的名字
  • cost() 方法就是计算费用,Drink 类中做成一个抽象方法
  • Decaf 就是单品咖啡, 继承Drink, 并实现cost
  • Espress && Milk 就是单品咖啡+调料, 这个组合很多
  • 问题:这样设计,会有很多类,当我们增加一个单品咖啡,或者一个新的调料, 类的数量就会倍增,就会出现类爆炸

优化方法一

前面分析到方案1因为咖啡单品+调料 组合会造成类的倍增,因此可以做改进,将调料内置到Drink类,这样就不会造成类数量过多。从而提高项目的维护性。

在这里插入图片描述

问题分析

  • 有效的缓解了类爆炸的问题
  • 但是在扩展的时候(新增或修改配料的时候)代码维护量过大,不仅要新增类,还需要修改drink

优化方法二:装饰者模式

装饰者模式定义

  • 装饰者模式:动态的将新功能附加到对象上在对象功能扩展方面,它比继承更有弹性,装饰者模式也体现了开闭原则(ocp)

装饰者模式原理

在这里插入图片描述

  1. 装饰者模式就像打包一个快递
  • 主体:比如:陶瓷、衣服 (Component) // 被装饰者
  • 包装:比如:报纸填充、塑料泡沫、纸板、木板(Decorator)
  1. Component 主体:比如类似前面的Drink

  2. ConcreteComponent和Decorator

    1. ConcreteComponent:具体的主体, 比如前面的各个单品咖啡
    2. Decorator: 装饰者,比如各调料
  3. 在如图的Component与ConcreteComponent之间,如果 ConcreteComponent类很多,还可以设计一个缓冲层,将共有的部分提取出来, 抽象层一个类

10.2 装饰者模式解决需求

使用装饰者模式设计的方案

在这里插入图片描述

说明

  • Drink 类就是前面说的抽象类, Component
  • ACoffee 就单品咖啡
  • Decorator 是一个装饰类,含有一个被装饰的对象(Drink obj)
  • Decorator 的cost 方法进行一个费用的叠加计算,递归的计算价格

代码示例

Drink类

/**
 * Drink类就是抽象的被装饰者,给具体的装饰者和被修饰者继承
 * @author cVzhanshi
 * @create 2023-09-05 15:16
 */
@Data
public abstract class Drink {
    // 描述
    private String des;

    // 价格
    private float price = 0.0f;

    // 计算费用的抽象方法
    // 子类来实现
    public abstract float cost();
}

Coffee类(缓冲层)

/**
 * 缓冲层
 * @author cVzhanshi
 * @create 2023-09-05 15:32
 */
public class Coffee extends Drink {
    @Override
    public float cost() {
        return super.getPrice();
    }
}

具体的被装饰者——各种咖啡

/**
 * @author cVzhanshi
 * @create 2023-09-05 15:33
 */
public class BCoffee extends Coffee {
    public BCoffee() {
        setDes("BCoffee");
        setPrice(3.0f);
    }
}

/**
 * @author cVzhanshi
 * @create 2023-09-05 15:33
 */
public class ACoffee extends Coffee {
    public ACoffee() {
        setDes("ACoffee");
        setPrice(5.0f);
    }
}

Decorator装饰类

/**
 * Decorator 是一个装饰类,含有一个被装饰的对象(Drink obj)
 * @author cVzhanshi
 * @create 2023-09-05 15:59
 */
public class Decorator extends Drink{

    private Drink coffee;

    public Decorator(Drink coffee) {
        this.coffee = coffee;
    }

    @Override
    public float cost() {
        // 调料的价格 + coffee的总价格
        return coffee.cost() + super.getPrice();
    }

    @Override
    public String getDes() {
        return super.getDes() + " || " + coffee.getDes();
    }
}

具体的装饰类

/**
 * @author cVzhanshi
 * @create 2023-09-05 16:09
 */
public class Milk extends Decorator {
    public Milk(Drink coffee) {
        super(coffee);
        setPrice(1.0f);
        setDes(" 牛奶 ");
    }
}

/**
 * @author cVzhanshi
 * @create 2023-09-05 16:10
 */
public class Soy extends Decorator {
    public Soy(Drink coffee) {
        super(coffee);
        setDes(" 糖 ");
        setPrice(9.0f);
    }
}

制作咖啡(允许效果)

/**
 * @author cVzhanshi
 * @create 2023-09-05 16:11
 */
public class CoffeeBar {
    public static void main(String[] args) {
        // 装饰者模式下订一份A咖啡 + 一份糖 + 两份牛奶

        // 1、点一份a咖啡
        Drink aCoffee = new ACoffee();

        System.out.println("费用1 = " + aCoffee.cost());
        System.out.println("描述 = " + aCoffee.getDes());

        // 2、a咖啡加入一份糖
        aCoffee = new Soy(aCoffee);

        System.out.println("aCoffee 加入一份糖 费用 = " + aCoffee.cost());
        System.out.println("aCoffee 加入一份糖 描述 = " + aCoffee.getDes());

        //2、再加入两份份牛奶
        aCoffee = new Milk(aCoffee);
        System.out.println("aCoffee 加入一份糖 一份牛奶 费用 = " + aCoffee.cost());
        System.out.println("aCoffee 加入一份糖 一份牛奶 描述 = " + aCoffee.getDes());
        aCoffee = new Milk(aCoffee);
        System.out.println("aCoffee 加入一份糖 两份牛奶 费用 = " + aCoffee.cost());
        System.out.println("aCoffee 加入一份糖 两份牛奶 描述 = " + aCoffee.getDes());
    }
}

//   允许效果
//        费用1 = 5.0
//        描述 = ACoffee
//        aCoffee 加入一份糖 费用 = 14.0
//        aCoffee 加入一份糖 描述 =  糖  || ACoffee
//        aCoffee 加入一份糖 一份牛奶 费用 = 15.0
//        aCoffee 加入一份糖 一份牛奶 描述 =  牛奶  ||  糖  || ACoffee
//        aCoffee 加入一份糖 两份牛奶 费用 = 16.0
//        aCoffee 加入一份糖 两份牛奶 描述 =  牛奶  ||  牛奶  ||  糖  || ACoffee

10.3 装饰者模式在JDK中的应用

通过阅读源码可知jdk中的io结构就是一个很典型的装饰者模式

结构如下:
在这里插入图片描述

  1. InputStream 是抽象类,类似我们前面讲的 Drink
public abstract class InputStream implements Closeable {
  // ....
}
  1. FilelnputStrearm 是 InputStream 子类,是具体的被装饰者类似我们前面的ACoffee、BCofee

  2. FilterInputStream 是InputStream 子类:类似我们前面的 Decorator 修饰者

public class FilterInputStream extends InputStream {
  /**
  FilterinputStream 类有 protected volatile InputStream in;即含被装饰者 
   * The input stream to be filtered.
   */
  protected volatile InputStream in;
  // ...
  }
  1. DatalnputStream 是 FilterinputStrcam 子类,具体的修饰者,类似前面的 Milk,Soy 等

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

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

相关文章

业务安全及案例实战

文章目录 业务安全1. 业务安全概述1.1 业务安全现状1.1.1 业务逻辑漏洞1.1.2 黑客攻击目标 2. 业务安全测试2.1 业务安全测试流程2.1.1 测试准备2.1.2 业务调研2.1.3 业务建模2.1.4 业务流程梳理2.1.5 业务风险点识别2.1.6 开展测试2.1.7 撰写报告 3. 业务安全经典场景3.1 业务…

conda的使用教程

conda的介绍 简单来说,conda软件就是来管理包的软件。以Python为例,在实际生活中,我们要处理多个不同的项目,因此,要安装不同的项目所需要的包,为了管理方便,conda就是用来打理不同项目的包&…

软考高级架构师下篇-13云原生架构设计理论与实践

目录 1. 考情分析2. 云原生架构内涵3. 云原生架构相关技术4. 前文回顾1. 考情分析 软考你报名了吗?下半年再来卷一个证书吧 本节主要学习云原生架构设计理论与实践。根据考试大纲,本小时知识点会涉及单选题型(约占2~4分)、案例题(25分)和论文题,本小时节内容偏重于方法…

自动化测试面试常见技术题目

1:一行代码实现1--100之和 print(sum(list(range(1,101)))) 2:如何在一个函数内部修改全局变量 global  修改全局变量 局部作用域只能调用全局作用域的变量,但是不熊修改全局作用域的变量,如果想要修改全局作用域的变量需要gl…

2023年中国信通院铸基计划“文本图像篡改检测系统技术规范”研讨会成功召开

2023年中国信通院铸基计划“文本图像篡改检测系统技术规范”(简称“规范”)研讨会于2023年8月16日在中国信息通信研究院成功召开,来自中国信息通信研究院、上海合合信息科技股份有限公司(简称“合合信息”)、华南理工大…

【文末送书】全栈开发流程——后端连接数据源(二)

前言 「作者主页」:雪碧有白泡泡 「个人网站」:雪碧的个人网站 「推荐专栏」: ★java一站式服务 ★ ★ React从入门到精通★ ★前端炫酷代码分享 ★ ★ 从0到英雄,vue成神之路★ ★ uniapp-从构建到提升★ ★ 从0到英雄&#xff…

【C#项目实战】控制台游戏 勇士斗恶龙(2)——游戏场景的设置以及玩家战斗逻辑

君兮_的个人主页 即使走的再远,也勿忘启程时的初心 C/C 游戏开发 Hello,米娜桑们,这里是君兮_,最近开始正式的步入学习游戏开发的正轨,想要通过写博客的方式来分享自己学到的知识和经验,这就是开设本专栏的目的。希望…

【web开发】2、css基础

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 一、CSS是什么?二、使用步骤2.1.css的存放位置2.2.选择器2.3.常用CSS样式介绍与示例 一、CSS是什么? 层叠样式表(英文全称:Casc…

vue2+element-ui批量导入方法并判断上传的文件是否为xls或xlsx

业务需求: 代码结构: <el-dialogtitle"批量导入":close-on-click-modal"true"close"close()":visible"true"width"35%":center"true"><div class"el-dialog-div"><!-- 头部区域布局 -…

万物互联:软件与硬件的协同之道

在当今数字化时代&#xff0c;我们身边的一切似乎都与计算机和互联网有关。从智能手机到智能家居设备&#xff0c;从自动驾驶汽车到工业生产线&#xff0c;无论我们走到哪里&#xff0c;都能看到软件和硬件的协同作用。本文将探讨这种协同作用&#xff0c;解释软件和硬件如何相…

R语言机器学习之caret包详解(二:模型的训练以及调参)

R语言机器学习之caret包详解&#xff08;二&#xff1a;模型的训练以及调参&#xff09; 前言caret包模型调优的策略、示例、以及一些小tips 前言 在之前的博客中我们详细介绍过了数据的拆分策略、各种数据处理的方法、各种交叉验证的方法&#xff0c;并且以示例介绍了R函数cre…

传统文化,运用好了,能生财!

千年以来&#xff0c;五行是中华民族文化的瑰宝&#xff0c;历代先贤智慧的结晶。五行学说是中国传统文化重要组成部分&#xff0c;是中国古代哲学思想的重要内容。古人认为&#xff0c;万物由五种相关的基本物质木、火、土、 金、水构成&#xff0c;这就是五行。 五行思想强调…

NFTScan 浏览器再升级:优质数据服务新体验来袭

当前&#xff0c;高质量的 NFT 数据服务已成为区块链用户和开发者的必需。为满足用户数据需求&#xff0c;NFTScan 主站近日进行全面升级&#xff0c;优化了数据服务板块的页面结构&#xff0c;实现更清晰简洁的布局和交互。 NFTScan 的改版充分考虑用户和开发者的数据体验&am…

【python】TCP socket服务器 Demo

目录 一、单线程服务器 二、多线程服务器 三、多线程服务器&#xff08;发送和接收分离&#xff09; 一、单线程服务器 说明&#xff1a;只能连接一个客户端 import socket,binascii# 创建一个 TCP 套接字 server_socket socket.socket(socket.AF_INET, socket.SOCK_STRE…

【C++基础】4、变量

文章目录 【 1、变量的定义 】【 2、变量的声明 】示例 【 3、左值和右值 】 变量&#xff1a;相当于是程序可操作的数据存储区的名称。在 C 中&#xff0c;有多种变量类型可用于存储不同种类的数据。C 中每个变量都有指定的类型&#xff0c;类型决定了变量存储的大小和布局&am…

文末送书!谈谈原型模式在JAVA实战开发中的应用(附源码+面试题)

作者主页&#xff1a;Designer 小郑 作者简介&#xff1a;3年JAVA全栈开发经验&#xff0c;专注JAVA技术、系统定制、远程指导&#xff0c;致力于企业数字化转型&#xff0c;CSDN博客专家&#xff0c;蓝桥云课认证讲师。 本文讲解了 Java 设计模式中的原型模式&#xff0c;并给…

python3

#安装python3 brew install python3 看到下图表示安装python3成功: #将python3 加入环境变量 export PATH$PATH:/opt/homebrew/bin/#查看python 版本 python3 --version#查看pip 版本 pip3 --version#更新python源 pip3 config set global.index-url https://pypi.tuna.tsing…

免费电商api接口,分享给你「建议收藏」

API&#xff08;应用程序编程接口&#xff09; ​ API&#xff08;Application Programming Interface&#xff0c;应用程序接口&#xff09;是一些预先定义的函数&#xff0c;或指软件系统不同组成部分衔接的约定。目的是提供应用程序与开发人员基于某软件或硬件得以访问一组…

高忆管理;三天翻倍!地产“三仙”狂飙

9月6日&#xff0c;一度全部成为“仙股”的三家闻名香港上市房企&#xff0c;融创我国、我国恒大、碧桂园大涨。其间两家已顺利摘掉“仙股”帽子。 截至发稿&#xff0c;融创我国涨20.25%&#xff0c;报1.9港元/股&#xff0c;总市值报104亿港元&#xff0c;不到三个交易日股价…

[LeetCode周赛复盘] 第 361 场周赛20230906

[LeetCode周赛复盘] 第 361 场周赛20230906 一、本周周赛总结2843. 统计对称整数的数目1. 题目描述2. 思路分析3. 代码实现 2844. 生成特殊数字的最少操作1. 题目描述2. 思路分析3. 代码实现 2845. 统计趣味子数组的数目1. 题目描述2. 思路分析3. 代码实现 2846. 边权重均等查…