设计模式21-组合模式
写在前面
数据结构模式
常常有一些组件在内部具有特定的数据结构。如何让客户程序依赖这些特定的数据结构,将极大的破坏组件的复用。那么这个时候将这些特定数据结构封装在内部。在外部提供统一的接口来实现与特定数据结构无关的访问。是一种行之有效的解决方案。
典型模式
- 组合模式
- 迭代器模式
动机
- 软件在某些情况下,客户代码过多的依赖于对象容器复杂的内部实现结构,对象容器内部实现结构而非抽象接口的变化将引起客户代码的频繁变化。代码的维护性,扩展性等弊端。
- 那么如何将客户代码与复杂的对象容器结构进行解耦?让对象容器自己来实现自身的复杂结构。而使得客户代码就像处理简单对象一样来实现处理复杂的对象容器?
- 在软件开发中,有时我们需要处理树形结构的数据。例如,图形编辑器中一个复杂图形可以由多个简单图形(如线条、圆形、矩形等)组合而成。无论是单个简单图形还是复杂图形的组合,从操作上看,它们应当被视为一个整体。组合模式的动机是通过将对象组合成树形结构来表示“部分-整体”的层次结构,使得客户端可以一致地处理单个对象和组合对象。
定义与结构
定义
组合模式允许你将对象组合成树形结构来表示“部分-整体”的层次结构。组合模式使得客户端对单个对象和组合对象的使用具有一致性。
结构
这张图是一个UML(统一建模语言)类图,用于展示软件系统中类之间的结构和关系。通过图形化的方式描述了类的属性、操作(方法)以及类之间的继承、关联等关系。
主要类及其关系
-
Client(客户端):
- 继承自Component类。
- 表示一个使用组件的客户端实体。客户端通过继承Component类,获得了对子节点的操作能力,包括添加、删除和获取子节点。
-
Component(组件):
- 是一个抽象类,代表了一个具有子组件概念的通用组件。
- 它定义了三个操作(方法):
Add(Component)
: 添加一个子组件。Remove(Component)
: 移除一个子组件。GetChild(int)
: 根据索引获取子组件。
- 还有一个属性
children
,用于存储子组件的集合,尽管这个属性在图中没有明确标出,但根据UML的惯例和类的操作可以推断出来。
-
Leaf(叶子节点):
- 继承自Component类。
- 表示没有子节点的组件,即树的叶子。
- 它同样定义了操作列表,但这里特别指出了一个
forall g in children
的操作,这实际上是一个伪代码或注释,因为叶子节点没有子节点(children
为空或不存在),所以这个操作在叶子节点上下文中不适用。这里的展示可能只是为了强调Leaf类继承自Component类,并保留了Component的接口结构。
-
Composite(复合节点):
- 继承自Component类。
- 表示具有多个子组件的复合结构,如树中的非叶子节点。
- 它除了具有Component类定义的操作外,还特别指出了对子节点
g
的操作(g.Operation():
),这里g
代表了一个子组件的实例,这个注释或伪代码表明Composite类可以对其子节点执行某种操作,但没有具体说明是什么操作,这取决于实际的应用场景。
这张UML类图展示了一个典型的组合模式(Composite Pattern)的结构,其中Component
是一个抽象类,代表了一个具有共同接口的对象,这个接口允许在组件的单个对象和组合对象之间进行一致的操作。Client
类展示了如何使用这个结构,而Leaf
和Composite
类则分别代表了结构中的叶子节点和复合节点。通过这种方式,系统可以以统一的方式处理单个对象和组合对象,简化了客户端代码并提高了系统的可扩展性。
C++代码推导
以下是一个使用组合模式的C++代码示例,模拟一个文件系统,其中目录可以包含文件或其他子目录。
抽象组件类:
#include <iostream>
#include <vector>
#include <string>// 抽象组件类,表示文件系统的节点
class FileSystemComponent {
public:virtual void showDetails(int indent = 0) const = 0;virtual void add(FileSystemComponent* component) {throw std::runtime_error("Cannot add to a leaf component");}virtual void remove(FileSystemComponent* component) {throw std::runtime_error("Cannot remove from a leaf component");}virtual ~FileSystemComponent() = default;
};
叶子节点类(文件):
class File : public FileSystemComponent {
private:std::string name;public:File(const std::string& name) : name(name) {}void showDetails(int indent = 0) const override {std::cout << std::string(indent, ' ') << name << std::endl;}
};
组合节点类(目录):
class Directory : public FileSystemComponent {
private:std::string name;std::vector<FileSystemComponent*> components;public:Directory(const std::string& name) : name(name) {}void add(FileSystemComponent* component) override {components.push_back(component);}void remove(FileSystemComponent* component) override {components.erase(std::remove(components.begin(), components.end(), component), components.end());}void showDetails(int indent = 0) const override {std::cout << std::string(indent, ' ') << name << "/" << std::endl;for (const auto& component : components) {component->showDetails(indent + 2);}}~Directory() {for (auto component : components) {delete component;}}
};
客户端代码:
int main() {FileSystemComponent* rootDir = new Directory("root");FileSystemComponent* homeDir = new Directory("home");FileSystemComponent* userDir = new Directory("user");FileSystemComponent* file1 = new File("file1.txt");FileSystemComponent* file2 = new File("file2.txt");FileSystemComponent* file3 = new File("file3.txt");rootDir->add(homeDir);homeDir->add(userDir);userDir->add(file1);userDir->add(file2);homeDir->add(file3);rootDir->showDetails();delete rootDir;return 0;
}
运行结果:
root/home/user/file1.txtfile2.txtfile3.txt
优缺点
优点:
- 统一性:组合模式使得客户端可以一致地处理单个对象和组合对象,统一了对叶子节点和组合节点的操作。
- 灵活性:可以很方便地增加新的节点类型(如新的文件类型或目录类型),符合开闭原则。
- 简化客户端代码:客户端无需关心处理的是单个对象还是组合对象,减少了代码复杂性。
缺点:
- 复杂性:可能会导致系统中类的数量增加,特别是当需要支持复杂的树形结构时。
- 难以限制组合:在组合模式中,很难限制哪些组件可以组合在一起,容易导致不合理的组合结构。
应用场景
组合模式在以下场景中应用较多:
- 需要表示树形结构的场景:如文件系统、组织结构、UI组件树等。
- 需要统一处理单个对象和组合对象的场景:如图形编辑器中的简单图形和组合图形。
- 需要动态构建部分-整体结构的场景:如菜单和子菜单的构建,产品配置和子组件的构建。
总结
- 组合模式通过将对象组合成树形结构来表示“部分-整体”的层次结构,使得客户端可以一致地处理单个对象和组合对象。它在需要处理树形结构的数据时非常有效,能够简化客户端代码,并提供很好的扩展性。然而,由于可能引入更多的类,特别是当系统的组合结构复杂时,需要注意管理组合的复杂性。
- 组合模式采用树形结构来实现普遍存在的对象容器,从而将一对多的关系转化为一对一的关系。使得客户代码可以一致的复用处理对象和和对象容器。无需关心处女的是单个对象还是组合的对象容器
- 将客户代码与复杂的对象容器结构解耦是组合模式的核心思想。解耦之后,客户代码将与纯粹的抽象接口而非对象容器的内部时间结构发生依赖,从而更能应对变化。
- 组合模式在具体的实现中可以让父对象中的子对象反向追溯。如果富对线有频繁的便利需求,可以使用缓存技巧来改善效率。
补充
在组合模式中,叶子节点通常不需要实现(即重载)Add(Component)
, Remove(Component)
, GetChild(int)
这三个方法,因为叶子节点不包含子节点。这些方法主要用于组合节点(Composite)以便管理子节点。但有时,为了简化代码或提高灵活性,叶子节点也可能会实现这些方法。以下是叶子节点重载与不重载这三个方法的优缺点对比。
叶子节点不重载这三个方法
实现方式:
在叶子节点中,这些方法通常被声明但不实现(在C++中通常可以抛出异常或者是空实现)。叶子节点不需要管理子组件。
class Leaf : public Component {
public:void Add(Component* component) override {throw std::runtime_error("Leaf nodes do not support Add operation");}void Remove(Component* component) override {throw std::runtime_error("Leaf nodes do not support Remove operation");}Component* GetChild(int index) override {throw std::runtime_error("Leaf nodes do not support GetChild operation");}void Operation() override {// 具体叶子节点的操作实现}
};
优点:
- 清晰的语义:叶子节点明确不支持子节点管理操作,这使得代码的意图更加清晰,避免了误用。
- 更强的类型安全:由于明确抛出异常或不实现,可以在运行时捕捉到错误,而不是让无意义的操作通过。
- 符合职责分离原则:叶子节点只专注于具体操作,不需要处理与子节点相关的逻辑。
缺点:
- 客户端代码需要做额外的检查:客户端需要知道一个组件是否是叶子节点,以避免调用不支持的方法,可能增加了客户端的复杂性。
- 减少了一致性:对客户端来说,调用这些方法会抛出异常或导致错误,这可能会影响代码的一致性和简洁性。
叶子节点重载这三个方法
实现方式:
叶子节点实现(重载)这些方法,但不执行任何操作或返回特定值,如nullptr
。
class Leaf : public Component {
public:void Add(Component* component) override {// 叶子节点不支持添加操作,但实现了这个方法}void Remove(Component* component) override {// 叶子节点不支持移除操作,但实现了这个方法}Component* GetChild(int index) override {return nullptr; // 叶子节点没有子节点,返回空指针}void Operation() override {// 具体叶子节点的操作实现}
};
优点:
- 简化客户端代码:客户端代码不需要检查节点类型,可以统一调用
Add
,Remove
,GetChild
,简化了代码逻辑。 - 提高一致性:所有组件(叶子节点和组合节点)都实现了相同的接口,提供了一致的编程接口。
- 增加灵活性:在未来扩展时,如果叶子节点需要支持子节点管理,可以直接扩展已有方法。
缺点:
- 隐藏潜在错误:叶子节点实现了不应该执行的操作(如
Add
和Remove
),可能导致误用而不易发现。 - 不符合职责分离原则:叶子节点本不应该涉及子节点管理操作,实现这些方法可能违反单一职责原则。
- 占用资源:虽然通常影响很小,但实现这些无操作的方法也会占用一些资源(例如代码空间),特别是在资源受限的环境中。
结论
-
不重载方法的情况:适用于严格遵循职责分离原则的场景。通过不重载方法,明确区分了叶子节点和组合节点的职责,使得代码更清晰,类型安全性更高。这种方式适合对系统稳定性和安全性要求较高的场合,或在需要明确捕获误用场景的应用中使用。
-
重载方法的情况:适用于追求客户端代码简单性和一致性的场景。通过重载这些方法,客户端不需要区分叶子节点和组合节点,统一处理所有组件,减少了代码的复杂性。这种方式适合在系统中灵活性要求较高、且不易出错的场合。
综上,选择是否重载这些方法取决于具体应用的需求、开发团队的编码习惯和系统的复杂性。如果系统需要严格的职责区分和类型安全性,建议不重载这些方法;如果系统追求统一性和简洁性,可以考虑重载这些方法。
相关文章:

设计模式21-组合模式
设计模式21-组合模式(Composite Pattern) 写在前面 动机定义与结构定义结构主要类及其关系 C代码推导优缺点应用场景总结补充叶子节点不重载这三个方法叶子节点重载这三个方法结论 写在前面 数据结构模式 常常有一些组件在内部具有特定的数据结构。如何…...

如何选择深度学习的损失函数和激活函数
一概述 在深度学习中,损失函数(Loss Function)和激活函数(Activation Function)是两个至关重要的组件,它们共同影响着模型的训练效果和泛化能力。本文将简要介绍这两个概念,阐述选择它们的重要性…...

DATAX自定义KafkaWriter
因为datax目前不支持写入数据到kafka中,因此本文主要介绍如何基于DataX自定义KafkaWriter,用来同步数据到kafka中。本文偏向实战,datax插件开发理论宝典请参考官方文档: https://github.com/alibaba/DataX/blob/master/dataxPlug…...

Mybatis分页多表多条件查询
个人总结三种方式: Xml、queryWrapper、PageHelper第三方组件这三种方式进行查询; 方式一: xml中联表查询,在mapper中传参IPage<T>和条件Map(这里用map装参数)。 代码示例: Mapper层 M…...

SpringBoot快速入门(手动创建)
目录 案例:需求 步骤 1 创建Maven项目 2 导入SpringBoot起步依赖 3 定义Controller 4 编写引导类 案例:需求 搭建简单的SpringBoot工程,创建hello的类定义h1的方法,返回Hello SpringBoot! 步骤 1 创建Maven项目 大家&…...

C 408—《数据结构》算法题基础篇—数组(通俗易懂)
目录 Δ前言 一、数组的合并 0.题目: 1.算法设计思想: 2.C语言描述: 3.算法的时间和空间复杂度 : 二、数组元素的倒置 0.题目 : 1.算法设计思想 : 2.C语言描述 : 3.算法的时间和空间复杂度 : 三、数组中特定值元素的删除 0.题目 : …...

AI秘境-墨小黑奇遇记 - 初体验(一)
“怎么可能!”墨小黑盯着屏幕上的代码,整个人都不好了。调试了三遍,翻了几遍书,结果还是不对。就像你以为自己早起赶车,结果发现闹钟根本没响一样崩溃。 这是他第一次真正接触人工智能实战任务——实现一个简单的感知…...

文件IO813
标准IO文件定位: fseek函数: 功能:将stream流文件中的文件指针从whence位置开始偏移offset个字节的长度。 int fseek(FILE *stream , long offset, int whence); FILE *stream 指的是所需要定位的文件(文化定位前提是文件要被打…...

STP(生成树)的概述和工作原理
💝💝💝欢迎来到我的博客,很高兴能够在这里和您见面!希望您在这里可以感受到一份轻松愉快的氛围,不仅可以获得有趣的内容和知识,也可以畅所欲言、分享您的想法和见解。 推荐:Linux运维老纪的首页…...

从AGV到立库,物流自动化的更迭与未来
AGV叉车 随着柔性制造系统的广泛应用,小批量、多批次的生产需求不断增强,“订单导向”生产已经成为趋势。这也让越来越多的企业认识到,产线的智能设备导入只是第一步,要想达到生产效率的最优解,物流系统的再优化必须提…...

阴阳脚数码管
1.小故事 最近,我接到了一个既“清肺”又“烧脑”的新任务,设计一个低功耗蓝牙肺活量计。在这个项目中我们借鉴了一款蓝牙跳绳的硬件设计方案,特别是它的显示方案——数码管。 在电子工程领域,初学者往往从操作LED开始ÿ…...

【Vue3-Typescript】<script setup lang=“ts“> 使用 ref标签 怎么获取 refs子组件呢
注意:请确保子组件已经正确挂载,并且通过 defineExpose 暴露了您想要在父组件中访问的属性或方法 parent.vue <template><child ref"childRef"></child><button click"fun">点击父组件</button> &l…...

npm 超详细使用教程
文章目录 一、简介二、npm安装三、npm 的使用3.1 npm初始化项目3.2 安装包3.3 安装不同版本包3.4 避免系统权限3.5 更新包3.6 卸载包3.7 执行脚本3.8 pre- 和 post- 脚本3.9 npm link3.10 发布和卸载发布的包3.11 使用npm版本控制3.22 npm资源 四、总结 一、简介 npmÿ…...

TypeScript函数
函数 函数:复用代码块 函数可以不写返回值 调用函数-----函数名() function a(){console.log(无参函数); } a();需要再函数后,写上返回值类型 没有返回值 使用void function e():string{return 可乐 } console.log(我得到了e()); function d():void{console.l…...

中海油某海上平台轨道巡检机器人解决方案
配电房作为能源传输和分配的核心枢纽,其安全运行直接影响到企业的生产稳定性和安全性。对于中海油这样的大型能源企业,配电房的运行状况至关重要。然而,传统的人工巡检方式存在效率低、作业风险高、巡检误差大等问题。为提升巡检效率、降低安…...

【NXP-MCXA153】SPI驱动移植
介绍 SPI总线由摩托罗拉公司开发,是一种全双工同步串行总线,由四个IO口组成:CS、SCLK、MISO、MOSI;通常用于CPU和外设之间进行通信,常见的SPI总线设备有:TFT LCD、QSPI FLASH、时钟模块、IMU等;…...

Python if 编程题|Python一对一辅导教学
你好,我是悦创。 以下为 if 编程练习题: 1. 奇数乘积问题 题目描述: 编写一个程序,判断给定的两个整数是否都是奇数,如果是,返回它们的乘积;如果不是,返回它们的和。输入: num1, num2输出: n…...

机器学习——第十一章 特征选择与稀疏学习
11.1 子集搜索与评价 对一个学习任务来说,给定属性集,其中有些属性可能很关键、很有用,另一些属性则可能没什么用.我们将属性称为"特征" (feature) ,对当前学习任务有用的属性称为"相关特征" (relevant featu…...

花式表演无人机技术详解
花式表演无人机作为现代科技与艺术融合的典范,以其独特的飞行姿态、绚烂的灯光效果及精准的控制能力,在各类庆典、体育赛事、音乐会等合中展现出非凡的魅力。本文将从以下几个方面对花式表演无人机技术进行详细解析。 1. 三维建模与编程 在花式表演无人…...

服务器那点事--防火墙
Linux服务器那点事--防火墙 Ⅰ、开启关闭Ⅱ、放开端口 Ⅰ、开启关闭 禁止防火墙开机自启systemctl disable firewalld 关闭防火墙systemctl stop firewalld 查看防火墙状态systemctl status firewalldⅡ、放开端口 例如:放开3306端口 设置放开3306端口 [rootbpm2…...

C:每日一题:单身狗
一、题目: 在一个整型数组中,只有一个数字出现一次,其他数组都是成对出现的,请找出那个只出现一次的数字。 整型数组 int arr[ ] {1,1,2,2,3,4,4} 二、思路分析: 1.,明确目标,选择…...

SQL之使用存储过程循环插入数据
1、已经创建了任务日志表 CREATE TABLE t_task_log (id bigint NOT NULL AUTO_INCREMENT,task_id bigint NOT NULL COMMENT 任务ID,read_time bigint NOT NULL COMMENT 单位秒,读取耗时,write_time bigint NOT NULL COMMENT 单位秒,写入耗时,read_size …...

智慧楼宇公厕系统小程序,提高卫生间管理使用效率
在当今的智慧楼宇中,公厕系统的管理和使用效率成为了衡量楼宇品质的重要指标之一。智慧楼宇公厕系统小程序的出现,为解决这一问题带来了全新的思路和方法。 一、检查公厕环境数据 智慧公厕系统不仅关注如厕的基本需求,还注重提升如厕环境的质…...

深度剖析:云数据库与传统数据库的显著差异
【若您对以下内容感兴趣,欢迎关注或联系我们】 在当今数字化时代,数据库技术不断演进,云数据库和传统数据库作为两种主要的数据库类型,在多个方面存在明显区别。下面我们将深入探讨这些差异。 一、部署方式 云数据库࿱…...

# 利刃出鞘_Tomcat 核心原理解析(六)
利刃出鞘_Tomcat 核心原理解析(六) 一、Tomcat专题 - 内容 1、Web 应用配置 2、Tomcat 管理配置 3、JVM 配置 4、Tomcat 集群 5、Tomcat 安全 6、Tomcat 性能调优 7、Tomcat 附加功能。 二、Tomcat专题 - Web应用配置介绍 1、Web.xml 配置文件…...

双亲委派模型
优质博文:IT-BLOG-CN 虚拟机设计团队把类加载阶段中的 “通过一个类的全限定名来获取描述此类的二进制字节流” 这个动作放到 Java虚拟机外部去实现,以便应用程序自己决定如何去获取所需要的类。实现这个动作的代码模块称为“类加载器”。 从Java虚拟机…...

Linux下ETCD安装、配置、命令
目录 1. ETCD简介 2. ETCD的安装 2.1 准备环境 2.2 下载ETCD 2.3 解压和移动文件 2.4 验证安装 3. ETCD的配置 3.1 基本配置 3.2 配置文件 3.3 集群配置 4. ETCD的常用命令 4.1 插入键值对 4.2 读取键值对 4.3 删除键值对 4.4 监视键的变化 4.5 列出所有键值 …...

【QT】静态库与动态库
文章目录 开始之前一、静态库(static Library)定义使用场景特点程序示例 二、动态库(dynamic Library)定义使用场景特点。程序示例第二种调用 开始之前 测试环境:Qt 5.15.2 Based on Qt 6.4.3(MSVC 2019, x86_64) 操作系统:Windows11 专业版 编程语言&am…...

R的行和列命名和类型的转换
下面内容摘录自: 4章8节:用R做数据重塑,行列命名和数据类型转换-CSDN博客 欢迎订阅我们专栏 一、行和列命名 在数据科学和统计分析中,命名是组织和管理数据的一个重要部分。尤其是在处理复杂的多维数据集时,为行和列命…...

某通用系统0day审计过程
前言 代码审计篇章都是自己跟几个师傅们一起审计的1day或者0day(当然都是小公司较为简单),禁止未经允许进行转载,发布到博客的用意主要是想跟师傅们能够交流下审计的思路,毕竟审计的思路也是有说法的,或者是相互源码共享也OK&…...