java设计模式——装饰者模式

定义: 装饰者模式是一种结构型设计模式,它允许动态地给对象添加新的功能,而不会改变其原有的结构。与继承不同,装饰者模式通过组合而不是继承来扩展对象的功能,这样可以有效地避免类爆炸问题(多个子类的冗余)。

在装饰者模式中,通常有以下几个关键角色:

抽象组件:定义对象的接口,可以是接口或抽象类。具体组件和装饰者都实现或继承该组件。

具体组件:实现抽象组件接口的具体类,它是被装饰的对象。

装饰者:实现抽象组件接口的类,内部维护一个抽象组件的引用,用于对被装饰对象进行扩展。

具体装饰者:继承装饰者并扩展其功能,可以为被装饰对象动态添加新功能。

优点

灵活性:通过组合而不是继承来扩展对象的功能,可以在运行时选择不同的装饰者动态组合对象。

遵循开闭原则:装饰者模式允许对功能进行扩展,而无需修改现有的代码。

减少子类的冗余:避免类层次的复杂性和继承的弊端。

缺点

复杂性增加:使用装饰者模式会增加系统中类的数量和对象的层次,增加理解和调试的难度。

难以维护:多个装饰者叠加时,调试可能变得困难,因为可能需要跟踪多个装饰者的行为。

实现示例

假设我们有一个咖啡店系统,每种咖啡有不同的类型(如普通咖啡、加牛奶的咖啡、加糖的咖啡等)。我们想要通过装饰者模式来灵活地添加配料,而不是为每种组合创建不同的类。

1. 定义抽象组件 Beverage

// 抽象组件
public abstract class Beverage {// 每种饮料都有一个描述和一个价格protected String description = "Unknown Beverage";public String getDescription() {return description;}public abstract double cost();
}

2. 定义具体组件 Coffee

// 具体组件 - 咖啡
public class Coffee extends Beverage {public Coffee() {description = "Coffee";}@Overridepublic double cost() {return 5.0; // 基本咖啡的价格}
}

3. 定义装饰者 CondimentDecorator

// 抽象装饰者 - 调料装饰器
public abstract class CondimentDecorator extends Beverage {// 强制要求具体装饰者必须实现 getDescription 方法public abstract String getDescription();
}

4. 定义具体装饰者 MilkSugar

// 具体装饰者 - 牛奶
public class Milk extends CondimentDecorator {// 被装饰的对象Beverage beverage;public Milk(Beverage beverage) {this.beverage = beverage;}@Overridepublic String getDescription() {return beverage.getDescription() + ", Milk";}@Overridepublic double cost() {return 1.5 + beverage.cost(); // 牛奶的价格加上原始饮料的价格}
}// 具体装饰者 - 糖
public class Sugar extends CondimentDecorator {// 被装饰的对象Beverage beverage;public Sugar(Beverage beverage) {this.beverage = beverage;}@Overridepublic String getDescription() {return beverage.getDescription() + ", Sugar";}@Overridepublic double cost() {return 0.5 + beverage.cost(); // 糖的价格加上原始饮料的价格}
}

5. 测试装饰者模式

public class CoffeeShop {public static void main(String[] args) {// 创建一杯基本的咖啡Beverage beverage = new Coffee();System.out.println(beverage.getDescription() + " $" + beverage.cost());// 给咖啡加牛奶beverage = new Milk(beverage);System.out.println(beverage.getDescription() + " $" + beverage.cost());// 给咖啡加牛奶和糖beverage = new Sugar(beverage);System.out.println(beverage.getDescription() + " $" + beverage.cost());}
}

输出结果:

Coffee $5.0
Coffee, Milk $6.5
Coffee, Milk, Sugar $7.0

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

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

相关文章

动手学深度学习9.7. 序列到序列学习(seq2seq)-笔记练习(PyTorch)

本节课程地址:62 序列到序列学习(seq2seq)【动手学深度学习v2】_哔哩哔哩_bilibili 本节教材地址:9.7. 序列到序列学习(seq2seq) — 动手学深度学习 2.0.0 documentation 本节开源代码:...>…

pdf编辑软件有哪些?方便好用的pdf编辑软件分享

PDF文件因其跨平台、格式固定的特性,成为了工作、学习和生活中不可或缺的一部分。然而,随着需求的不断增加,仅仅阅读PDF文件已难以满足我们的需求,编辑、转换PDF文件成为了新的焦点,下面给大家分享几款方便好用的PDF编…

《Linux从小白到高手》综合应用篇:深入理解Linux常用关键内核参数及其调优

1. 题记 有关Linux关键内核参数的调整,我前面的调优文章其实就有涉及到,只是比较零散,本篇集中深入介绍Linux常用关键内核参数及其调优,Linux调优80%以上都涉及到内核的这些参数的调整。 2. 文件系统相关参数 fs.file-max 参数…

SpringBoot3 + MyBatisPlus 快速整合

一、前言 MyBatis 最佳搭档,只做增强不做改变,为简化开发、提高效率而生。 这个发展到目前阶段已经很成熟了,社区也比较活跃,可以放心使用。官网地址:https://baomidou.com 二、快速开始 引入依赖 这里我引入了核心…

stm32单片机个人学习笔记11(ADC模数转换器)

前言 本篇文章属于stm32单片机(以下简称单片机)的学习笔记,来源于B站教学视频。下面是这位up主的视频链接。本文为个人学习笔记,只能做参考,细节方面建议观看视频,肯定受益匪浅。 STM32入门教程-2023版 细…

Linux系列-Linux的常见指令(三)

🌈个人主页:羽晨同学 💫个人格言:“成为自己未来的主人~” mv 1.剪切文件,目录 2.重命名 首先,我们先来看第一个作用 假如说,我们原先存在一个hello.txt,我们如果想要将这个文件移动到其他的…

上拉电阻和下拉电阻在电路中的作用(一)

上拉电阻和下拉电阻在电路中的作用(一) 1.什么是上下拉电阻2.上下拉电阻的作用:2.1.维持输入引脚处于稳定状态。2.2.配合三极管和MOS进行电平转换电路设计2.3.OC、OD电路(Open Collector集电极开路、Open Drain漏电极开路&#xf…

什么是分库分表?为什么要分库分表?什么时候需要分库分表?怎么样拆分?(数据库分库分表详解)

文章目录 1、什么是分库分表?1.1、分库分表的概念1.2、分库分表的方式1.2.1、垂直分库1.2.2、垂直分表1.2.3、水平分库1.2.4、水平分表 2、为什么要分库分表?3、什么时候需要分库分表?4、分库分表的数据路由4.1、数据路由的目的4.2、数据路由…

class 9: vue.js 3 组件化基础(2)父子组件间通信

目录 父子组件之间的相互通信父组件传递数据给子组件Prop为字符串类型的数组Prop为对象类型 子组件传递数据给父组件 父子组件之间的相互通信 开发过程中,我们通常会将一个页面拆分成多个组件,然后将这些组件通过组合或者嵌套的方式构建页面。组件的嵌套…

2024开放原子开源生态大会 | 麒麟信安携手openEuler共建开源生态,共塑产业未来

9月25日-27日,由开放原子开源基金会主办的2024开放原子开源生态大会在北京开幕,大会以“开源赋能产业,生态共筑未来”为主题。工业和信息化部党组书记、部长金壮龙,北京市委副书记、市长殷勇,工业和信息化部总经济师、…

汇川机器人与PLC通信-ModbusTCP超详细案例

#SCARA机器人与H5UPLC通过ModbusTCP通信,HMI界面手动操作# 应用背景: 本项目案例部分软件界面已被更新,如机器人示教软件旧版本S01.19R03。但通信的原理基本一致,废话少说,我们直接上图。 一、PLC端配置 1.添加ROB通讯表(自定义),变量表内容包括ROB系统变量,IN区和…

Cadence元件A属性和B属性相互覆盖

最近在使用第三方插件集成到Cadence,协助导出BOM到平台上,方便对BOM进行管理和修改,结果因为属性A和属性B不相同,导致导出的BOM错误。如下图: ​​ 本来我们需要导出Q12,结果给我们导出了Q13,或者反之&…

基于opencv的人脸闭眼识别疲劳监测

1. 项目简介 本项目旨在实现基于眼部特征的眨眼检测,通过监测眼睛开闭状态来计算眨眼次数,从而应用于疲劳监测、注意力检测等场景。使用了面部特征点检测算法,以及眼部特征比率(EAR, Eye Aspect Ratio)来判断眼睛的闭…

Python 实现 excel 数据过滤

一、场景分析 假设有如下一份 excel 数据 shop.xlsx, 写一段 python 程序,实现对于车牌的分组数据过滤。 并以车牌为文件名,把店名输出到 车牌.txt 文件中。 比如 闽A.txt 文件内容为: 小林书店福州店1 小林书店福州店2 二、依赖安装 程序依…

【C++】拆分详解 - 模板

文章目录 一、泛型编程二、函数模板1. 概念2. 语法3. 函数模板的原理4. 函数模板的实例化5. 模板参数的匹配原则 三、类模板1. 语法2. 实例化 四、模板的特化1. 概念2. 函数模板特化3. 类模板特化3.1 全特化3.2 偏特化 / 半特化3.3 应用示例 4. 小结 五、模板的分离编译1. 分离…

Java:抽象类和接口

一.抽象类 1.抽象类概念和语法 ⨀概念: 在面向对象的概念中,所有的对象都是通过类来描绘的,但是并不是所有的类都是用来描绘对象的,如果一个类中没有包含足够的信息来描绘一个具体的对象,这样的类就是抽象类。 ⨀语…

JMeter使用不同方式传递接口参数

1、使用 HTTP 请求中的参数: 在 JMeter 的测试计划中,添加一个 "HTTP 请求" 元件。 在 "HTTP 请求" 元件的参数化选项中,可以添加参数的名称和值。可以手动输入参数,也可以使用变量来传递参数值。 如果要使…

Golang | Leetcode Golang题解之第497题非重叠矩形中的随机点

题目: 题解: type Solution struct {rects [][]intsum []int }func Constructor(rects [][]int) Solution {sum : make([]int, len(rects)1)for i, r : range rects {a, b, x, y : r[0], r[1], r[2], r[3]sum[i1] sum[i] (x-a1)*(y-b1)}return Sol…

自定义多级联动选择器指南(uni-app)

多端支持:可以运行在H5、APP、微信小程序还是支付宝小程序,都可以轻松使用改组件。自定义配置:您可以根据需要配置选择器的级数,使其适应不同的数据结构和用例。无限级联:此组件支持无限级联选择,使您能够创…

最好的ppt模板网站是哪个?做PPT不可错过的18个网站!

现在有很多PPT模板网站,但真正免费且高质量的不多,今天我就分享主流的国内外PPT模板下载网站,并且会详细分析这些网站的优缺点,这些网站都是基于个人实际使用经验的,免费站点会特别标注,让你可以放心下载&a…