news 2026/4/3 4:13:36

组合模式详解

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
组合模式详解

什么是组合模式?

组合模式(Composite Pattern),是一种结构型设计模式,这种模式将对象组合成树形结构,以表示部分--整体的层次关系,组合模式使得用户对单个对象和组合对象的使用具有一致性。

组合模式的结构

角色说明
Component抽象组件,定义叶子和容器的公共接口
Leaf叶子节点,实现 Component 的具体行为(如 File)
Composite容器节点,持有子组件列表,实现 Component 并提供 add/remove 等管理方法(如 Folder)

代码实现-文件系统示例

第一步:定义抽象组件FileSystemItem

public abstract class FileSystemItem { protected String name; public FileSystemItem(String name) { this.name = name; } /** * 显示文件或目录的信息 * @param indent 缩进级别,用于显示层级结构 */ public abstract void display(int indent); /** * 获取文件或目录的大小 * @return 文件或目录的大小(字节) */ public abstract long getSize(); // 工具方法:生成缩进 protected String getIndent(int level) { return " ".repeat(level); } }

第二步:实现叶子节点File

public class File extends FileSystemItem { private long size; public File(String name, long size) { super(name); this.size = size; } /** * 显示文件或目录的信息 * * @param indent 缩进级别,用于显示层级结构 */ @Override public void display(int indent) { System.out.println(getIndent(indent) + "📄 " + name + " (" + size + "KB)"); } /** * 获取文件或目录的大小 * * @return 文件或目录的大小(字节) */ @Override public long getSize() { return size; } }

第三步:实现容器节点Folder

public class Folder extends FileSystemItem { private List<FileSystemItem> children = new ArrayList<>(); public Folder(String name) { super(name); } /** * 显示文件或目录的信息 * * @param indent 缩进级别,用于显示层级结构 */ @Override public void display(int indent) { System.out.println(getIndent(indent) + "📁 " + name); for (FileSystemItem child : children) { child.display(indent + 1); // 递归显示子项 } } // 管理子组件的方法(仅 Composite 需要) public void add(FileSystemItem item) { children.add(item); } public void remove(FileSystemItem item) { children.remove(item); } public List<FileSystemItem> getChildren() { return Collections.unmodifiableList(children); } /** * 获取文件或目录的大小 * * @return 文件或目录的大小(字节) */ @Override public long getSize() { return children.stream() .mapToLong(FileSystemItem::getSize) .sum(); } }

第四步:客户端使用

public class CompositeDemo { public static void main(String[] args) { // 构建树形结构 Folder project = new Folder("MyProject"); Folder docs = new Folder("docs"); docs.add(new File("requirements.md", 10)); docs.add(new File("design.png", 500)); Folder src = new Folder("src"); src.add(new File("Main.java", 20)); src.add(new File("Utils.java", 15)); project.add(docs); project.add(src); project.add(new File("README.md", 5)); // 统一操作!无需区分 File 或 Folder project.display(0); System.out.println("\n总大小: " + project.getSize() + " KB"); } }

运行输出:

两种组合模式变体

类型特点优缺点
透明式(Transparent)Component 接口中声明所有方法(包括add,remove客户端完全透明,但 Leaf 必须抛出UnsupportedOperationException
安全式(Safe)只在 Composite 中定义管理子节点的方法类型安全,但客户端需知道对象是否为 Composite(如需调用add

我们上面的例子采用的是安全式,更符合 Java 的类型安全理念。

组合模式的优点

  • 简化客户端代码:统一接口,无需判断类型。
  • 易于扩展:新增组件类型不影响现有逻辑。
  • 天然支持递归操作:非常适合树形数据结构。

典型应用场景

  • 🖥️GUI 组件树:Swing/AWT 中的ContainerComponent
  • 🌐DOM 树操作:HTML 元素的父子结构
  • 🏢组织架构:公司 → 部门 → 员工
  • 🎮游戏对象层级:场景 → 角色 → 武器/特效
  • 📂菜单系统:主菜单 → 子菜单 → 菜单项

总结

组合模式通过抽象出统一的行为接口,巧妙地将“个体”与“整体”统一处理,极大提升了代码的可读性、可维护性和可扩展性

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

AI编程:Trae CN用户规则和项目规则定义分享

诸神缄默不语-个人技术博文与视频目录 文章目录1. 用户规则2. 项目规则1. 网站前端3. 参考资料1. 用户规则 位置在C:\Users\user_name.trae-cn\user_rules.md下&#xff0c;Trae CN官方引导&#xff1a; 参考提示词&#xff1a; **核心理念与原则** > **简洁至上**&#…

作者头像 李华
网站建设 2026/3/9 23:30:09

基于SpringBoot+Vue的超市食品安全管理系统设计与实现

前言 &#x1f31e;博主介绍&#xff1a;✌CSDN特邀作者、全栈领域优质创作者、10年IT从业经验、码云/掘金/知乎/B站/华为云/阿里云等平台优质作者、专注于Java、小程序/APP、python、大数据等技术领域和毕业项目实战&#xff0c;以及程序定制化开发、文档编写、答疑辅导等。✌…

作者头像 李华
网站建设 2026/4/2 1:56:28

USB数据线/串口线---无法识别问题全解@

问题一&#xff1a;手机用USB数据线连接电脑&#xff0c;可充电但电脑读取不到手机怎么办&#xff1f; 通常&#xff0c;当用USB数据线连接手机和电脑后&#xff0c;在电脑桌面的右下角会有连接提示&#xff0c;并在电脑桌面的中间弹出查看文件的对话框。 &#xff08;有些线是…

作者头像 李华
网站建设 2026/3/30 18:30:56

《用好 Pandas,轻松驾驭大数据:高效处理技巧与实战指南》

《用好 Pandas&#xff0c;轻松驾驭大数据&#xff1a;高效处理技巧与实战指南》 在数据驱动的时代&#xff0c;Python 的 pandas 库几乎是每位数据分析师、工程师和科学家的“标配”。它以灵活的数据结构&#xff08;如 DataFrame 和 Series&#xff09;、丰富的操作接口和优雅…

作者头像 李华
网站建设 2026/4/2 13:36:42

刷新页面不等于新页面,一次由油猴脚本不执行解决的暗坑

复述 我打算在B站编写一个油猴脚本来提供便利。于是直接写好脚本后刷新页面&#xff0c;发现不运行。打开油猴脚本的扩展提示“该脚本未执行”。 经过好一段时间的痛苦排查也没有发现到底是反爬机制还是脚本错误导致的问题&#xff0c;我无意中打开了新的B站页面&#xff0c;脚…

作者头像 李华
网站建设 2026/3/30 21:05:54

C++笔记:流式异步日志库

综合我之前学过的异步日志库&#xff0c;流的缓冲区以及TensorRT里sample的日志设计。总结出了一套流式异步日志。 参考文章&#xff1a; C笔记&#xff1a;实现小型日志系统-CSDN博客 TensorRT笔记&#xff08;2&#xff09;&#xff1a;解析样例中Logger日志类的设计-CSDN…

作者头像 李华