news 2026/3/10 11:54:55

设计模式[9]——装饰器模式一分钟彻底说清楚

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
设计模式[9]——装饰器模式一分钟彻底说清楚

设计模式[9]——装饰器模式一分钟彻底说透

一句话定义

在不修改原有对象的前提下,运行时动态、透明地给对象层层添加额外行为,保持接口不变。

软件领域真实例子:网络数据流处理(超级常见!)

场景:你有一个基本的网络 Socket 发送数据,但实际项目中经常需要层层叠加功能

  • 先加密(Encryption)
  • 再压缩(Compression)
  • 再加日志记录(Logging)
  • 最后才真正发送

客户端代码只关心stream->write(data),完全不知道中间加了多少层处理!

不用装饰器会怎样?继承爆炸:

classEncryptedSocket:publicSocket{...}classEncryptedCompressedSocket:publicEncryptedSocket{...}classEncryptedCompressedLoggedSocket:publicEncryptedCompressedSocket{...}// 3种功能 → 8种子类!明天加“校验和”功能?直接崩溃。
很多人困惑:这不就是普通的继承多态吗?

错得离谱!下面这张表10秒彻底撕开:

项目普通继承多态(你觉得像的)装饰器模式(真相)
添加行为时机静态:编译时固定在子类里动态:运行时随意叠加、顺序随意、数量随意
类数量功能组合爆炸(2^n 个子类)只有 n + 1 个类(基础 + n种装饰)
灵活性想临时加/减一层功能?不可能运行时随意包装:想加双重加密?包两层就行
关系本质is-a(EncryptedSocket “是一种” Socket)has-a(装饰器“持有”一个被装饰的对象)
软件真实场景固定类型:TcpSocket、UdpSocket临时增强:加密流、压缩流、日志流、缓冲流
口号“焊死在子类里”“运行时层层叠加,想加就加,想拆就拆”

狠话:继承是生孩子——生完就定型;装饰器是穿外套——想穿几件、哪件先穿都行,还能脱!

真实软件代码(现代 C++20,网络流处理)
#include<iostream>#include<memory>#include<string>#include<vector>usingnamespacestd;// 1. 统一数据流接口classDataStream{public:virtual~DataStream()=default;virtualvoidwrite(constvector<uint8_t>&data)=0;virtualstringname()const=0;// 用于调试描述};// 2. 基础流(真正干活的核心)classTcpSocketStream:publicDataStream{public:voidwrite(constvector<uint8_t>&data)override{cout<<"[TcpSocket] 真正发送数据,长度: "<<data.size()<<" 字节"<<endl;}stringname()constoverride{return"TCP Socket";}};// 3. 装饰器基类(关键:持有一个流指针)classStreamDecorator:publicDataStream{protected:unique_ptr<DataStream>stream;// ← 包装链的核心public:explicitStreamDecorator(unique_ptr<DataStream>s):stream(move(s)){}stringname()constoverride{returnstream->name();// 默认转发,子类可追加}};// 4. 具体装饰器(真实功能)classLoggingDecorator:publicStreamDecorator{public:explicitLoggingDecorator(unique_ptr<DataStream>s):StreamDecorator(move(s)){}voidwrite(constvector<uint8_t>&data)override{cout<<"[Logging] 记录写操作,数据大小: "<<data.size()<<endl;stream->write(data);// 转发给下一层}stringname()constoverride{returnstream->name()+" + Logging";}};classCompressionDecorator:publicStreamDecorator{public:explicitCompressionDecorator(unique_ptr<DataStream>s):StreamDecorator(move(s)){}voidwrite(constvector<uint8_t>&data)override{cout<<"[Compression] 正在压缩数据... (原大小 "<<data.size()<<" -> 压缩后 "<<data.size()/2<<")"<<endl;autocompressed=compress(data);// 模拟压缩stream->write(compressed);}stringname()constoverride{returnstream->name()+" + Compression";}private:vector<uint8_t>compress(constvector<uint8_t>&d){return{1,2,3};}// 简化};classEncryptionDecorator:publicStreamDecorator{public:explicitEncryptionDecorator(unique_ptr<DataStream>s):StreamDecorator(move(s)){}voidwrite(constvector<uint8_t>&data)override{cout<<"[Encryption] 正在AES加密数据..."<<endl;autoencrypted=encrypt(data);// 模拟加密stream->write(encrypted);}stringname()constoverride{returnstream->name()+" + Encryption";}private:vector<uint8_t>encrypt(constvector<uint8_t>&d){return{9,9,9};}// 简化};
客户端:运行时随意组装流
intmain(){vector<uint8_t>payload={1,2,3,4,5,6,7,8,9,0};// 场景1:生产环境 - 全功能autostream=make_unique<TcpSocketStream>();stream=make_unique<LoggingDecorator>(move(stream));stream=make_unique<CompressionDecorator>(move(stream));stream=make_unique<EncryptionDecorator>(move(stream));cout<<"当前流: "<<stream->name()<<endl;// TCP Socket + Logging + Compression + Encryptionstream->write(payload);// 场景2:调试环境 - 只加日志,不加密不压缩autodebugStream=make_unique<TcpSocketStream>();debugStream=make_unique<LoggingDecorator>(move(debugStream));debugStream->write(payload);// 场景3:想双重加密?轻松!autosecureStream=make_unique<TcpSocketStream>();secureStream=make_unique<EncryptionDecorator>(move(secureStream));secureStream=make_unique<EncryptionDecorator>(move(secureStream));// 双重加密!secureStream->write(payload);}
C++ 真实项目里到处都是装饰器
  • 标准库std::ostreamstd::ofstreamstd::stringstream(层层包装)
  • Boost.Asio:socket → ssl::stream → buffered_stream
  • Qt:QWidget → QScrollArea(滚动条装饰) → CustomBorderDecorator
  • 游戏网络:Packet → CRCDecorator → EncryptDecorator → CompressDecorator → Send
  • 日志库:Logger → FileWriter → AsyncDecorator → RotateDecorator
终极口诀(软件人专属)

“功能运行时叠,层层包装不继承;
接口不变真优雅,加密压缩随便配!”

刻在DNA里的一句话

当你在软件中需要“运行时给一个流/对象临时叠加多个处理行为”(日志、加密、压缩、缓冲等),且组合方式多变时,
别用继承爆炸,用装饰器——层层包装,接口统一,灵活到极致!

现在,装饰器模式在纯软件场景下彻底说透了吧?
这才是程序员每天都在用的真实模式!

下一期要外观模式(Facade),它专门治“子系统太复杂”的病!

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/3/7 14:06:29

UDP 协议详解与 Qt 实战应用

引言&#xff1a; https://github.com/0voice UDP&#xff08;User Datagram Protocol&#xff0c;用户数据报协议&#xff09;是 TCP/IP 协议簇中传输层的核心协议之一&#xff0c;与 TCP 协议共同承担着端到端的数据传输任务。相较于 TCP 的面向连接、可靠传输特性&#xf…

作者头像 李华
网站建设 2026/2/28 16:23:51

为什么你的MAUI应用上线就崩溃?99%开发者忽略的测试盲区曝光

第一章&#xff1a;为什么你的MAUI应用上线就崩溃&#xff1f;99%开发者忽略的测试盲区曝光 在.NET MAUI开发中&#xff0c;许多开发者发现应用在本地调试时运行正常&#xff0c;但一旦发布到生产环境便频繁崩溃。问题根源往往隐藏在被忽视的测试盲区中——尤其是平台特定行为、…

作者头像 李华
网站建设 2026/2/28 16:23:49

Unity学习 2Dadventure 4

一 UI - 创建人物状态栏创建canvas切换自己创建的控制器切换比例通过Alt控制位置裁切ui设置 自动裁切并手动处理&#xff0c;然后再为需要的命名设置为固定比例&#xff0c;修改大小和位置复制一份并且注意层级&#xff0c;调整大小&#xff0c;修改填充设置这里就是一些个性化…

作者头像 李华
网站建设 2026/3/2 6:22:38

9、计算机数字表示、架构与内存管理知识详解

计算机数字表示、架构与内存管理知识详解 1. 二进制与数字表示 在计算机领域,数字的表示方式多种多样。二进制是计算机最基础的表示形式,例如二进制数 110110 对应的十进制数是 54。 1.1 二进制算术 二进制加法是二进制算术的基本操作,用于两个二进制数相加。 1.2 数字…

作者头像 李华
网站建设 2026/3/8 13:14:37

PHP安全审计实战:从0到1构建医疗数据防护体系(含真实案例)

第一章&#xff1a;医疗数据安全面临的挑战与PHP审计必要性 在数字化转型加速的背景下&#xff0c;医疗行业越来越多地依赖基于Web的信息系统来管理患者记录、诊断结果和治疗方案。这些系统中&#xff0c;PHP作为广泛应用的服务器端脚本语言&#xff0c;常用于构建医院管理平台…

作者头像 李华
网站建设 2026/3/4 15:19:31

扩展QDK有多难?3个关键接口让你快速上手

第一章&#xff1a;扩展QDK有多难&#xff1f;从认知到实践的跨越量子开发工具包&#xff08;Quantum Development Kit, QDK&#xff09;由微软提供&#xff0c;旨在简化量子算法的开发与仿真。尽管其抽象层次较高&#xff0c;但要真正扩展QDK以支持自定义量子操作或集成新后端…

作者头像 李华