0

0

c++如何实现装饰模式_c++动态给对象添加功能【实战】

尼克

尼克

发布时间:2026-02-09 04:19:22

|

515人浏览过

|

来源于php中文网

原创

装饰模式的核心是组合而非继承,所有装饰器与被装饰对象实现同一抽象接口(如IDrawable),通过持有接口指针转发调用并增强逻辑;必须使用std::unique_ptr管理所有权、保持const正确性、避免虚调用开销。

c++如何实现装饰模式_c++动态给对象添加功能【实战】

装饰模式的核心不是继承,而是组合与接口一致

直接用继承扩展功能在 C++ 里看似简单,但会导致类爆炸、无法运行时选择行为、违反开闭原则。装饰模式的关键在于:所有装饰器和被装饰对象都实现同一抽象接口(比如 IDrawable),而装饰器内部持有一个指向该接口的指针(或引用),把调用“转发”过去——再在前后加自己的逻辑。

常见错误是让装饰器继承具体类(如 RedShape : public Circle),这会锁死类型,失去装饰任意 IDrawable 的能力。

  • 必须定义纯虚基类(如 IDrawable),所有实体类和装饰器都 public virtual 继承它
  • 装饰器构造函数接收 IDrawable*std::unique_ptr,避免裸指针生命周期失控
  • 不要在装饰器里重写所有函数——只重写需要增强的那几个,其余直接委托

用 unique_ptr 管理装饰链,避免内存泄漏和悬挂指针

手动 new / delete 构建装饰链(比如 new BorderDecorator(new ColorDecorator(new Circle())))极易出错:谁负责释放?异常发生时怎么办?C++11 后标准做法是用 std::unique_ptr 自动管理所有权。

示例:构建一个带边框+红色填充的圆形

立即学习C++免费学习笔记(深入)”;

Flowith
Flowith

一款GPT4驱动的节点式 AI 创作工具

下载
auto shape = std::make_unique();
shape = std::make_unique(std::move(shape), "red");
shape = std::make_unique(std::move(shape), 2);
shape->draw(); // 输出:Drawing red Circle with border width 2
  • std::move 是关键:每次包装都移交所有权,避免拷贝和重复释放
  • 装饰器的成员变量应为 std::unique_ptr,而非裸指针
  • 如果需要共享底层对象(比如多个装饰器引用同一原始对象),才考虑 std::shared_ptr,但要警惕循环引用

装饰器不能改变被装饰对象的 const 正确性

如果原始对象是 const IDrawable&,而你的装饰器 draw() 函数没加 const 修饰,编译就会失败——因为委托调用需要匹配 const 限定符。

典型错误写法:void draw() override { /* ... */ } → 无法接受 const 对象;正确写法:void draw() const override { component_->draw(); }

  • 所有接口函数声明必须带 const(如果它们不修改逻辑状态)
  • 装饰器内部的 component_ 成员也应是 std::unique_ptr 或通过 const 引用传递
  • 若装饰器自身需维护可变状态(如计数器),用 mutable 修饰该成员,保持接口 const

避免过度装饰带来的虚函数调用开销和调试困难

每层装饰器都是一次虚函数调用跳转。5 层嵌套意味着 draw() 调用要经过 5 次 vtable 查找——对高频调用路径(如渲染循环)可能成为瓶颈。更隐蔽的问题是:堆跟踪里全是 ColorDecorator::draw → BorderDecorator::draw → ...,掩盖了真正业务逻辑的位置。

  • 优先用编译期方案(如模板策略、CRTP)替代运行期装饰,当行为组合固定且数量有限时
  • [[likely]] / [[unlikely]] 标注分支预测,但无法消除虚调用本身
  • 加日志时别只打 “decorator called”,要输出 typeid(*component_).name() 和当前装饰类型,否则查链式调用像盲人摸象

最常被忽略的一点:装饰器的析构顺序和构造顺序相反,但如果你在某层装饰器里做了资源申请(比如 OpenGL texture 绑定),必须确保它在对应 draw() 之后才释放——而不是依赖析构时机。这要求把“后置清理”逻辑显式塞进 draw() 尾部,而不是放在 destructor 里。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

541

2023.09.20

javascriptvoid(o)怎么解决
javascriptvoid(o)怎么解决

javascriptvoid(o)的解决办法:1、检查语法错误;2、确保正确的执行环境;3、检查其他代码的冲突;4、使用事件委托;5、使用其他绑定方式;6、检查外部资源等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

180

2023.11.23

java中void的含义
java中void的含义

本专题整合了Java中void的相关内容,阅读专题下面的文章了解更多详细内容。

107

2025.11.27

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1344

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

277

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

2201

2025.12.29

java接口相关教程
java接口相关教程

本专题整合了java接口相关内容,阅读专题下面的文章了解更多详细内容。

34

2026.01.19

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

404

2023.07.18

Golang处理数据库错误教程合集
Golang处理数据库错误教程合集

本专题整合了Golang数据库错误处理方法、技巧、管理策略相关内容,阅读专题下面的文章了解更多详细内容。

98

2026.02.06

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
国外Web开发全栈课程全集
国外Web开发全栈课程全集

共12课时 | 1万人学习

进程与SOCKET
进程与SOCKET

共6课时 | 0.4万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号