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

设计模式每日硬核训练 Day 14:组合模式(Composite Pattern)完整讲解与实战应用

🔄 回顾 Day 13:桥接模式小结

在 Day 13 中,我们学习了桥接模式(Bridge Pattern):

  • 用于将“抽象”与“实现”分离,适用于双维度变化场景(如图形类型 × 渲染方式)。
  • 它强调组合替代继承,解决类爆炸问题,提升系统可扩展性。

今天我们进入一个构建层级结构的重要模式——组合模式(Composite Pattern)

组合模式的目标是:让你以一致的方式对待单个对象和对象集合(树形结构)


一、组合模式的核心动机

✅ 什么是组合模式?

组合模式用于构建树状结构的对象系统,例如:

  • 文件夹包含文件和子文件夹
  • UI 容器包含多个控件
  • 公司组织结构:员工 ← 部门 ← 公司

核心哲学:

将对象组成树形结构,客户端可以“统一操作”叶子节点与中间节点。


二、UML 结构图

+----------------+
|   Component    |<------------------------------+
+----------------+                               |
| +operation()   |                               |
+----------------+                               |/\                                        |/  \                                       |
+-------------------+     +---------------------+ |
|  Leaf             |     |  Composite           | |
+-------------------+     +---------------------+ |
| +operation()      |     | +add(Component*)     | || +remove(Component*)  | || +operation()         | |+---------------------+ |

在这里插入图片描述


三、角色说明

角色职责说明
Component抽象类,统一接口
Leaf叶子节点,实现具体功能,不含子节点
Composite组合节点,内部维护子组件列表

四、C++ 实现:文件系统结构

✅ 抽象组件接口

class FileSystemNode {
public:virtual void display(int depth = 0) = 0;virtual ~FileSystemNode() = default;
};

✅ 叶子节点:文件

class File : public FileSystemNode {std::string name_;
public:File(const std::string& name) : name_(name) {}void display(int depth = 0) override {std::cout << std::string(depth, '-') << name_ << std::endl;}
};

✅ 组合节点:文件夹

class Directory : public FileSystemNode {std::string name_;std::vector<std::unique_ptr<FileSystemNode>> children_;public:Directory(const std::string& name) : name_(name) {}void add(std::unique_ptr<FileSystemNode> node) {children_.emplace_back(std::move(node));}void display(int depth = 0) override {std::cout << std::string(depth, '-') << name_ << "/" << std::endl;for (const auto& child : children_) {child->display(depth + 2);}}
};

✅ 使用示例

int main() {auto root = std::make_unique<Directory>("root");root->add(std::make_unique<File>("file1.txt"));auto subDir = std::make_unique<Directory>("subdir");subDir->add(std::make_unique<File>("file2.txt"));subDir->add(std::make_unique<File>("file3.txt"));root->add(std::move(subDir));root->display();return 0;
}

输出:

root/
--file1.txt
--subdir/
----file2.txt
----file3.txt

五、组合模式适用场景

场景对象树结构说明
操作系统文件系统文件 + 文件夹,操作接口统一
图形界面控件窗口、容器、按钮、文本框构成控件树
公司组织架构CEO → 部门主管 → 员工
报表结构层级表头、表体、表尾、字段
HTML DOM 树节点 + 元素 + 属性

六、优点与缺点总结

✅ 优点:

  • 统一接口,客户端无差别调用
  • 树结构天然适合层次建模
  • 扩展方便,添加新节点只需实现 Component

❗ 缺点:

  • 违背接口隔离原则:叶子节点和组合节点共用接口,部分函数空实现
  • 调试复杂,结构越深越难定位问题

七、与装饰器 / 责任链等模式对比

模式核心区别类似点
Composite结构树形,有聚合子对象Component 接口统一
Decorator功能增强,包裹单一对象接口一致、动态组合
Chain责任链传递,节点决定是否继续传递多节点连接,共同参与处理

八、面试回答模板

“在我们的配置中心中,使用组合模式构建配置节点树,既可以是叶子属性(字段),也可以是组合节点(嵌套组)。所有节点都继承自统一接口,使我们可以用递归统一地遍历配置结构、序列化、验证,代码简洁且扩展性好。”

✅ 建议突出树形结构、递归遍历、统一调用等优势。


九、记忆口诀

“树形结构走统一,组合调用不分离;叶子整体皆一类,层层嵌套递归易。”


十、明日预告:Day 15

享元模式(Flyweight Pattern):通过共享技术减少对象数量,提升内存利用效率。

http://www.xdnf.cn/news/4303.html

相关文章:

  • 基于Django实现的图书分析大屏系统项目
  • Linux 常用命令总结
  • NLP高频面试题(四十六)——Transformer 架构中的位置编码及其演化详解
  • MCP和A2A是什么?
  • FreeRTOS事件标志组
  • 【Linux】第八章 监控和管理Linux进程
  • 关于Diamond机械手的运动学与动力学的推导
  • 【力扣刷题】49字母异位词分组,不用哈希,c语言实现
  • 《AI大模型应知应会100篇》第22篇:系统提示词(System Prompt)设计与优化
  • 基础知识 - 结构体
  • 首席人工智能官(Chief Artificial Intelligence Officer,CAIO)的详细解析
  • 从“链主”到“全链”:供应链数字化转型的底层逻辑
  • 智能sc一面
  • 【cocos creator 3.x】cocos creator2.x项目升级3.x项目改动点
  • 士兵乱斗(贪心)
  • 前端api(请求后端)简易template
  • Python高级爬虫之JS逆向+安卓逆向1.5节: 控制结构
  • docker harbor私有仓库登录报错
  • Ubuntu利用docker搭建Java相关环境问题记录
  • 如何有效防止服务器被攻击
  • 在激烈竞争下B端HMI设计怎样打造独特用户体验?
  • 数组理论基础
  • 从GPT到Gemini 大模型进化史
  • ADVB发送器设计
  • Matter如何终结智能家居生态割据,重构你的居住体验?
  • 随手笔记-python-opencv 读取图像的顺序 与pytorch处理图像的顺序
  • Mysql的安装
  • Java面试(2025)—— Spring
  • FPGA入门学习Day1——设计一个DDS信号发生器
  • opencv HSV的具体描述