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

从Java的JDK源码中学设计模式之装饰器模式

装饰器模式是一种极具弹性的结构型设计模式,它允许我们通过组合的方式动态扩展对象功能而无需修改原有结构。本文将通过JDK源码中的实际应用和通俗易懂的代码示例,带你深入了解这一强大模式的精髓。

装饰器模式核心原理

装饰器模式的核心思想:在原有对象外面"包装"一层新功能,同时保持与被装饰对象相同的接口。它能够:

  1. 在不改变对象的前提下增强功能
  2. 避免因过度继承导致类爆炸
  3. 支持运行时动态添加功能
  4. 组合替代继承提高灵活性

Java IO包中的装饰器模式实战

让我们深入JDK源码(Java 17),看看java.io包如何完美应用装饰器模式:

import java.io.*;public class DecoratorInJavaIO {public static void main(String[] args) throws IOException {// 基础数据类型装饰DataInputStream dataInput = new DataInputStream(new BufferedInputStream(new FileInputStream("data.bin")));// 字符编码转换装饰BufferedReader reader = new BufferedReader(new InputStreamReader(new FileInputStream("text.txt"), "UTF-8"));// 动态添加行号功能LineNumberReader lineReader = new LineNumberReader(reader);// 动态添加大小写转换装饰器UpperCaseReader upperReader = new UpperCaseReader(lineReader);String line;while ((line = upperReader.readLine()) != null) {int num = upperReader.getLineNumber();System.out.println("Line " + num + ": " + line);}}
}// 自定义装饰器:将内容转为大写
class UpperCaseReader extends FilterReader {protected UpperCaseReader(Reader in) {super(in);}@Overridepublic int read() throws IOException {int c = super.read();return (c == -1) ? c : Character.toUpperCase(c);}@Overridepublic int read(char[] cbuf, int off, int len) throws IOException {int n = super.read(cbuf, off, len);for (int i = off; i < off + n; i++) {cbuf[i] = Character.toUpperCase(cbuf[i]);}return n;}// 增强功能:提供读取整行的方法public String readLine() throws IOException {char[] buffer = new char[1024];int pos = 0;int c;while ((c = read()) != -1) {if (c == '\n') break;buffer[pos++] = (char)c;}if (pos == 0 && c == -1) return null;return new String(buffer, 0, pos);}
}

在上述代码中:

  1. 我们使用Java IO的核心装饰器(BufferedInputStream, InputStreamReader
  2. 展示了装饰器链式组合的强大功能
  3. 创建了自定义的装饰器UpperCaseReader来扩展原有功能

装饰器模式结构解析

下面使用Mermaid工具展示装饰器模式的类图结构:

持有引用
Component
+operation() : void
ConcreteComponent
+operation() : void
Decorator
-component: Component
+Decorator(Component)
+operation() : void
ConcreteDecoratorA
+operation() : void
+addedBehavior() : void
ConcreteDecoratorB
+addedState: String
+operation() : void

图中关键角色:

  1. Component: 被装饰对象的公共接口(如Java的InputStream)
  2. ConcreteComponent: 基础实现(如FileInputStream)
  3. Decorator: 装饰器抽象层(如FilterInputStream)
  4. ConcreteDecorator: 具体装饰器实现(如BufferedInputStream)

JDK中装饰器模式实现原理

分析java.io.FilterInputStream源码:

public class FilterInputStream extends InputStream {protected volatile InputStream in;protected FilterInputStream(InputStream in) {this.in = in;}public int read() throws IOException {return in.read();}// 所有方法都委托给in对象public int read(byte[] b) throws IOException {return read(b, 0, b.length);}public int read(byte[] b, int off, int len) throws IOException {return in.read(b, off, len);}// 其他方法...
}

在JDK实现中:

  1. 所有具体装饰器都继承自FilterInputStream
  2. 每个装饰器持有底层InputStream的引用
  3. 基础方法直接委托给底层流
  4. 需要增强的方法被重写(如BufferedInputStream缓冲功能)

装饰器模式 vs 继承

特点装饰器模式继承
扩展方式运行时编译时
组合方式对象组合类继承
灵活性高(动态组合)低(静态绑定)
功能叠加线性添加只能单一路径
修改风险无(不修改原类)需要修改类层次

装饰器模式的典型应用场景

  1. 输入/输出流处理:Java IO/NIO中的流装饰
  2. Servlet API:HttpServletRequestWrapper装饰请求
  3. Collections工具类:unmodifiableXXX创建不可变视图
  4. JavaFX应用:Node对象的多种样式装饰
// Java集合框架中的装饰器应用
List<String> origin = new ArrayList<>();
List<String> safeList = Collections.checkedList(origin, String.class);
List<String> unmodifiable = Collections.unmodifiableList(origin);

装饰器模式的优点与局限

核心优势:

  • 符合开闭原则:扩展不修改
  • 职责明确:小类单一职责
  • 动态组合:运行时装配功能
  • 避免类爆炸:取代多层继承结构

潜在缺点:

  • 过度使用导致结构复杂
  • 调试困难(调用链路深)
  • 小对象数量可能增加

总结与最佳实践

装饰器模式在Java核心库特别是IO系统中发挥了至关重要的作用。它通过优雅的包装机制,实现了功能的动态组合,避免了传统继承的固有问题。

使用建议:

  1. 当需要动态、透明地添加职责时
  2. 当不适合使用子类扩展时
  3. 当目标可能有多种不同组合时
  4. 当需要保持被装饰对象的接口纯净时

掌握装饰器模式将使你的设计更具弹性,帮助创建更灵活、可扩展的系统架构。同时也要注意避免过度装饰导致的复杂性,在恰当的场景发挥其最大价值。

设计思想的精髓: 组合优于继承,封闭修改打开扩展,通过对象包装而非类继承来实现功能增强!

相关文章:

从Java的JDK源码中学设计模式之装饰器模式

装饰器模式是一种极具弹性的结构型设计模式&#xff0c;它允许我们通过组合的方式动态扩展对象功能而无需修改原有结构。本文将通过JDK源码中的实际应用和通俗易懂的代码示例&#xff0c;带你深入了解这一强大模式的精髓。 装饰器模式核心原理 装饰器模式的核心思想&#xff…...

鸿蒙电脑会在国内逐渐取代windows电脑吗?

点击上方关注 “终端研发部” 设为“星标”&#xff0c;和你一起掌握更多数据库知识 10年内应该不会 用Windows、MacOS操作系统的后果是你的个人信息可能会被美国FBI看到&#xff0c;但绝大多数人的信息FBI没兴趣去看 你用某家公司的电脑系统,那就得做好被某些人监视的下场,相信…...

持续领跑中国异地组网路由器市场,贝锐蒲公英再次登顶销量榜首

作为国产远程连接SaaS服务的创领者&#xff0c;贝锐持续引领行业发展&#xff0c;旗下贝锐蒲公英异地组网路由器&#xff0c;凭借出色的技术实力和市场表现&#xff0c;斩获2024年线上电商平台市场销量份额中国第一的佳绩&#xff0c;充分彰显了其在网络解决方案与异地组网领域…...

Spring AI 系列3: Promt提示词

一、Promt提示词 Promt提示是引导 AI 模型生成特定输出的输入&#xff0c; 提示的设计和措辞会显著影响模型的响应。 在 Spring AI 中与 AI 模型交互的最低层级&#xff0c;处理提示有点类似于在 Spring MVC 中管理”视图”。 这涉及创建带有动态内容占位符的大段文本。 这些占…...

Nginx 的配置文件

核心概念&#xff1a; 指令 (Directives): 配置文件的基本构建块。每条指令以分号 ; 结束。指令可以设置参数&#xff08;如 worker_processes auto;&#xff09;。 上下文 (Contexts): 指令被组织在特定的块&#xff08;用花括号 {} 包围&#xff09;中&#xff0c;称为上下文…...

Redis:安装与常用命令

&#x1f308; 个人主页&#xff1a;Zfox_ &#x1f525; 系列专栏&#xff1a;Redis &#x1f525; 安装 Redis 使⽤apt安装 apt install redis -y⽀持远程连接 修改 /etc/redis/redis.conf 修改 bind 127.0.0.1 为 bind 0.0.0.0 修改 protected-mode yes 为 protected-mo…...

[原创](Windows使用技巧): Windwos11如何设置局域网共享访问? (多图详解)

[作者] 常用网名: 猪头三 出生日期: 1981.XX.XX 企鹅交流: 643439947 个人网站: 80x86汇编小站 编程生涯: 2001年~至今[共24年] 职业生涯: 22年 开发语言: C/C++、80x86ASM、Object Pascal、Objective-C、C#、R、Python、PHP、Perl、 开发工具: Visual Studio、Delphi、XCode、…...

Mac 芯片系列 安装cocoapod 教程

安装声明&#xff1a; 本人是在搭梯子的环境下安装成功&#xff0c;前提是必须安装好安装homebrew环境。 1.检测rudy的源 2.查看源(目的:检测rudy的源) gem sources - l 3.移除源(目的:移除rudy自带的源) gem sources --remove https://rubygems.org/ 4.更换源(目的:替换成国…...

智启未来:AI重构制造业供应链的五大革命性突破

一、需求预测&#xff1a;让供应链“未卜先知” 1.1 从经验判断到数据预言 传统供应链依赖人工分析历史数据&#xff0c;但面对市场波动、设备突发故障等不确定性&#xff0c;往往反应滞后。AI通过整合工业物联网&#xff08;IIoT&#xff09;传感器数据、生产排程、供应商交…...

Linux进程间通信----简易进程池实现

进程池的模拟实现 1.进程池的原理&#xff1a; 是什么 进程池是一种多进程编程模式&#xff0c;核心思想是先创建好一定数量的子进程用作当作资源&#xff0c;这些进程可以帮助完成任务并且重复利用&#xff0c;避免频繁的进程的创建和销毁的开销。 下面我们举例子来帮助理…...

解锁Java多级缓存:性能飞升的秘密武器

一、引言 文末有彩蛋 在当今高并发、低延迟的应用场景中&#xff0c;传统的单级缓存策略往往难以满足性能需求。随着系统规模扩大&#xff0c;数据访问的瓶颈逐渐显现&#xff0c;如何高效管理缓存成为开发者面临的重大挑战。多级缓存架构应运而生&#xff0c;通过分层缓存设…...

(纳芯微)NCA9548- DTSXR 具有复位功能的八通道 I²C 开关、所有I/O端子均可承受5.5V输入电压

深圳市润泽芯电子有限公司 推荐NOVOSENSE(纳芯微)品牌 NCA9548- DTSXR TSSOP-24封装 NCA9548- DTSXR 具有复位功能的八通道 IC 开关、所有I/O端子均可承受5.5V输入电压 产品描述 NCA9548是通过I2C总线控制的八路双向转换开关。 SCL / SDA上行数据分散到八对下行数据或通道。…...

013旅游网站设计技术详解:打造一站式旅游服务平台

旅游网站设计技术详解&#xff1a;打造一站式旅游服务平台 在互联网与旅游业深度融合的时代&#xff0c;旅游网站成为人们规划行程、预订服务的重要工具。一个功能完备的旅游网站&#xff0c;通过用户管理、订单管理等核心模块&#xff0c;实现用户与管理员的高效交互。本文将…...

2024 CKA模拟系统制作 | Step-By-Step | 12、题目搭建-创建多容器Pod

目录 免费获取题库配套 CKA_v1.31_模拟系统 一、题目 二、考点分析 1. 多容器 Pod 的理解 2. YAML 配置规范 3. 镜像版本控制 三、考点详细讲解 1. 多容器 Pod 的工作原理 2. 容器端口冲突处理 3. 资源隔离机制 四、实验环境搭建步骤 总结 免费获取题库配套 CKA_v…...

优化 Spring Boot API 性能:利用 GZIP 压缩处理大型有效载荷

引言 在构建需要处理和传输大量数据的API服务时&#xff0c;响应时间是一个关键的性能指标。一个常见的场景是&#xff0c;即使后端逻辑和数据库查询已得到充分优化&#xff0c;当API端点返回大型数据集&#xff08;例如&#xff0c;数千条记录的列表&#xff09;时&#xff0…...

PostgreSQL 修改表结构卡住不动

[] 查找卡住的进程 ID&#xff08;PID&#xff09; -- 查看当前所有数据库连接及进程信息 SELECTpid,usename,query,age(clock_timestamp(), query_start) AS query_duration FROMpg_stat_activity WHEREquery LIKE %ALTER TABLE%; -- 过滤出正在执行 ALTER TABLE 的语句今天遇…...

【C盘瘦身】给DevEco Studio中HarmonyOSEmulator(鸿蒙模拟器)换个地方,一键移动给C盘瘦身

文章目录 一、HarmonyOSEmulator的安装路径二、修改路径 一、HarmonyOSEmulator的安装路径 之前安装了华为的DevEco Studio&#xff0c;当时没注意&#xff0c;后来C盘告急&#xff0c;想着估计是鸿蒙的模拟器占用空间比较大&#xff0c;一检查还真是躺在C盘。路径如下&#x…...

AutoCompose - 携程自动编排【开源】

AutoCompose - 携程自动编排【开源】 AutoCompose是一款单事件驱动&#xff08;无状态&#xff09;的流程引擎。使用本框架&#xff0c;能够轻松实现复杂服务的自动化编排【零配置、零编码】&#xff0c;能够显著提高开发维护效率。支持同步编程、异步编程&#xff08;已支持Co…...

mybatis和hibernate区别

MyBatis 和 Hibernate 都是 Java 生态中主流的持久层框架&#xff0c;但设计理念和适用场景有显著区别。以下是核心对比&#xff1a; 1. 本质区别 特性HibernateMyBatis框架类型全自动 ORM&#xff08;对象关系映射&#xff09;框架半自动 SQL 映射框架核心思想对象优先&#…...

ORACLE 缺失 OracleDBConsoleorcl服务导致https://xxx:port/em 不能访问

这个原因是&#xff0c;操作过一下 ORCL的服务配置变更导致的。 再PATH中添加个环境变量&#xff0c;路径如下 管理员权限运行cmd 等待创建完成 大概3分钟 查看服务 点击第一个访问&#xff0c;下图登录后的截图...

unix/linux source 命令,其历史争议、兼容性、生态、未来展望

现在把目光投向unix/linux source命令的历史争议、兼容性、生态和未来展望,这能让我们更全面地理解一个技术点在更广阔的图景中所处的位置。 一、历史争议与设计权衡 虽然 source (或 .) 命令功能强大且不可或缺,但在其发展和使用过程中,也存在一些微妙的争议或设计上的权衡…...

day42 简单CNN

目录 一、从图像分类任务谈起 二、CNN架构解剖实验室 2.1 卷积层&#xff1a;空间特征的魔法师 2.2 归一化层&#xff1a;加速收敛的隐形推手 2.3 激活函数&#xff1a;非线性的灵魂 三、工程实践避坑指南 3.1 数据增强工程 3.2 调度器工程实战 四、典型问题排查手册 …...

VScode自动添加指定内容

在 VS Code 中&#xff0c;可以通过配置 用户代码片段&#xff08;User Snippets&#xff09; 或使用 文件模板扩展 来实现新建指定文件类型时自动添加指定内容。以下是具体方法&#xff1a; 方法 1&#xff1a;使用 VS Code 内置的「用户代码片段」 适用场景&#xff1a;适用…...

Ubuntu 22.04 安装 Nacos 记录

Ubuntu 22.04 安装 Nacos 记录 本文记录了在 Ubuntu 22.04 系统上安装 Nacos 的完整过程&#xff0c;适用于本地测试或生产部署的基础搭建。 一、官方资源 官网下载地址&#xff1a;https://nacos.io/download/nacos-server/官网文档&#xff1a;https://nacos.io/docs/lates…...

终极陷阱:Java序列化漏洞的内爆原理与防御体系重建

引言&#xff1a;被遗忘的后门 2019年Equifax公司因Java反序列化漏洞导致1.43亿用户数据泄露&#xff0c;最终以7亿美元达成和解。令人震惊的是&#xff0c;问题源头竟是一个简单的序列化接口&#xff1a; public class UserSession implements Serializable {private String…...

Git 中移除已追踪的文件

你已经成功提交了部分文件到 Git&#xff0c;但 sqlserver/data/ 目录下的一些日志文件&#xff08;如 .xel 和 machine-key&#xff09;仍然被追踪或未被忽略。你想 彻底忽略整个 sqlserver/data/* 目录下的所有内容。 ✅ 目标 让 Git 忽略以下路径&#xff1a; sqlserver/d…...

相机--RGBD相机

教程 分类原理和标定 原理 视频总结 双目相机和RGBD相机原理 作用 RGBD相机RGB相机深度&#xff1b; RGB-D相机同时获取两种核心数据&#xff1a;RGB彩色图像和深度图像&#xff08;Depth Image&#xff09;。 1. RGB彩色图像 数据格式&#xff1a; 标准三通道矩阵&#…...

Flask中secret_key设置解析

app.secret_key os.urandom(24) 在 Flask 中&#xff0c;app.secret_key os.urandom(24) 这行代码用于生成并设置一个安全的随机密钥&#xff08;Secret Key&#xff09;&#xff0c;这是 Flask 应用中非常重要的配置之一。以下是详细解析&#xff1a; ​1. app.secret_key …...

事件不触发、交互失效?基于 WebDebugX 的移动端事件调试实战总结

在移动端开发中&#xff0c;“点击没反应”“滑动卡住”“长按无效”等事件类问题时常困扰开发者。这类问题不仅和逻辑代码有关&#xff0c;更常见的是出现在浏览器事件模型与设备行为之间的不一致&#xff0c;特别是在 WebView 环境下尤为显著。 本文结合多个真实案例&#x…...

记一次idea中lombok无法使用的解决方案

在注解处理器下&#xff0c;一般 Default 为“启用注解处理”和“从项目类路径获取处理器”&#xff0c;但是我的项目中的为选择“处理器路径”&#xff0c;导致了无法识别lombok&#xff0c;因此&#xff0c;需要改为使用“从项目类路径获取处理器”这个选项。如下图所示&…...