组合模式详解(Java)

一、组合模式基本概念
1.1 定义与类型

组合模式是一种结构型设计模式,它通过将对象组织成树形结构,来表示“部分-整体”的层次关系。这种模式使得客户端可以一致地对待单个对象和组合对象,从而简化了客户端代码的复杂性。组合模式的核心在于定义了一个抽象组件角色,这个角色既可以代表叶子节点,也可以代表容器节点,每个节点都可以包含子节点。

在组合模式中,对象被组织成一种树形结构,这种结构允许对象以层次化的方式组合,形成一个整体。树形结构的顶端是根节点,根节点下面可以有多个子节点,这些子节点可以是叶子节点,也可以是其他容器节点。这种结构允许我们以一致的方式来处理单个对象和组合对象,使得客户端代码更加简洁和统一。

// 抽象组件角色
public interface Component {void operation();void add(Component component);void remove(Component component);Component getChild(int index);
}// 叶子节点角色
public class Leaf implements Component {@Overridepublic void operation() {// 具体操作}@Overridepublic void add(Component component) {// 叶子节点不能添加子节点throw new UnsupportedOperationException();}@Overridepublic void remove(Component component) {// 叶子节点不能移除子节点throw new UnsupportedOperationException();}@Overridepublic Component getChild(int index) {// 叶子节点没有子节点throw new IndexOutOfBoundsException();}
}// 容器节点角色
public class Composite implements Component {private List<Component> children = new ArrayList<>();@Overridepublic void operation() {// 具体操作for (Component component : children) {component.operation();}}@Overridepublic void add(Component component) {children.add(component);}@Overridepublic void remove(Component component) {children.remove(component);}@Overridepublic Component getChild(int index) {return children.get(index);}
}
1.2 组合模式的目的

组合模式的主要目的是使得客户端可以一致地对待单个对象和组合对象。通过这种方式,客户端无需关心它处理的是单个对象还是一组对象,这大大简化了客户端的代码复杂性。例如,在一个文件系统中,文件和文件夹可以被视为组合模式中的对象,客户端可以以相同的方式操作文件和文件夹,而无需知道它们的具体类型。

此外,组合模式通过提供一个统一的接口,使得新增节点类型变得非常容易,而无需修改现有的代码。这符合设计模式中的开闭原则,即在增加新功能时,无需修改现有代码。通过组合模式,我们可以轻松地在树形结构中添加或删除节点,而不会影响其他部分的代码。这种灵活性使得组合模式在处理复杂层次结构时显得尤为强大。

二、组合模式的结构
2.1 抽象组件角色(Component)

抽象组件角色是组合模式中的核心,它定义了树结构中所有对象的公共接口和方法。这个角色通常是一个接口或抽象类,它声明了所有子类都必须实现的方法,如添加子节点、删除子节点等操作。通过抽象组件角色,客户端可以一致地对待所有对象,无论是叶子节点还是容器节点。

抽象组件角色的主要职责是提供一个统一的操作接口,使得客户端可以在不知道具体对象类型的情况下进行操作。例如,在一个图形用户界面中,抽象组件角色可以定义绘制、移动和缩放等操作,无论是按钮、文本框还是复合面板,都必须实现这些方法。这样,客户端就可以以相同的方式处理所有这些组件,而无需关心它们的具体类型。

public interface Component {void operation();void add(Component component);void remove(Component component);Component getChild(int index);
}
2.2 叶子节点角色(Leaf)

叶子节点角色表示树结构中的末端节点,它不包含

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/diannao/70157.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

LabVIEW危化品仓库的安全监测系统

本案例展示了基于LabVIEW平台设计的危化品仓库安全监测系统&#xff0c;结合ZigBee无线通信技术、485串口通讯技术和传感器技术&#xff0c;实现了对危化品仓库的实时无线监测。该系统不仅能提高安全性&#xff0c;还能大幅提升工作效率&#xff0c;确保危化品仓库的安全运营。…

【私人笔记】Web前端

Vue专题 vue3 vue3 页面路径前面添加目录 - 路由base设置 - vite设置base https://mbd.baidu.com/ma/s/XdDrePju 修改vite.config.js export default defineConfig({base: /your-directory/,// 其他配置... }); vue2 uniapp 【持续更新】uni-app学习笔记_uniapp快速复制一…

数仓搭建:DWB层(基础数据层)

维度退化: 通过减少表的数量和提高数据的冗余来优化查询性能。 在维度退化中&#xff0c;相关的维度数据被合并到一个宽表中&#xff0c;减少了查询时需要进行的表连接操作。例如&#xff0c;在销售数据仓库中&#xff0c;客户信息、产品信息和时间信息等维度可能会被合并到一…

【Linux】进程间通信——进程池

文章目录 进程池什么进程池进程池的作用 用代码模拟进程池管道信息任务类InitProcesspool()DisPatchTasks()任务的执行逻辑&#xff08;Work&#xff09;CleanProcessPool() 封装main.ccChannel.hppProcessPool.hppTask.hppMakefile 总结总结 进程池 什么进程池 进程池&#…

13-跳跃游戏 II

给定一个长度为 n 的 0 索引整数数组 nums。初始位置为 nums[0]。 每个元素 nums[i] 表示从索引 i 向后跳转的最大长度。换句话说&#xff0c;如果你在 nums[i] 处&#xff0c;你可以跳转到任意 nums[i j] 处: 0 < j < nums[i] i j < n 返回到达 nums[n - 1] 的最…

Qt的QToolBox的使用

QToolBox 是 Qt 框架中的一个控件&#xff0c;用于创建一个可折叠的“工具箱”界面&#xff08;类似 Windows 资源管理器的侧边栏&#xff09;。每个子项可以展开或折叠&#xff0c;适合用于分组显示多个功能模块。以下是其基本用法和示例&#xff1a; 1. 基本用法 创建并添加…

《DeepSeek 一站式工作生活 AI 助手》

最近国产AI工具DeepSeek在全球火出圈&#xff0c;登顶多个国家应用商店&#xff0c;下载量一路飙升。这匹AI “黑马” 到底凭什么征服全球用户&#xff1f;让我们全方位解锁DeepSeek——从基础入门到高阶玩法&#xff0c;从实用技巧到隐藏功能。 DeepSeek是一款功能强大的国产A…

Java中CompletableFuture异步工具类

参考&#xff1a;CompletableFuture 详解 | JavaGuide 实际项目中&#xff0c;一个接口可能需要同时获取多种不同的数据&#xff0c;然后再汇总返回&#xff0c;举个例子&#xff1a;用户请求获取订单信息&#xff0c;可能需要同时获取用户信息、商品详情、物流信息、等数据。…

Oracle Rac 多路径链路不稳定引发IO降速-光弱

一、背景 今天突然被异地的同事拉来开远程会议&#xff0c;会议内容是开发反馈每天9点左右有个sqlldr 命令的脚本调用突然执行很慢&#xff0c;以前几秒的导入操作现在需要30-60s左右&#xff0c;而且数据量基本相同。 二、分析 1&#xff09;、查看ASH报告 从报告上确认是数…

哈希表-两个数的交集

代码随想录-刷题笔记 349. 两个数组的交集 - 力扣&#xff08;LeetCode&#xff09; 内容: 集合的使用 , 重复的数剔除掉&#xff0c;剩下的即为交集&#xff0c;最后加入数组即可。 class Solution {public int[] intersection(int[] nums1, int[] nums2) {Set<Integer…

[JVM篇]分代垃圾回收

分代垃圾回收 分代收集法是目前大部分 JVM 所采用的方法&#xff0c;其核心思想是根据对象存活的不同生命周期将内存划分为不同的域&#xff0c;一般情况下将 GC 堆划分为老生代(Tenured/Old Generation)和新生代(Young Generation)。老生代的特点是每次垃圾回收时只有少量对象…

汉诺塔问题详解:递归与分治的经典案例

嘿&#xff0c;小伙伴们&#xff01;今天我可算撞见了个超有意思的东西&#xff0c;就是那大名鼎鼎的汉诺塔问题&#xff01;我这好奇心一下子就被勾起来了&#xff0c;迫不及待地想深挖一下&#xff0c;然后把那些好玩的、烧脑的、让人拍案叫绝的解题思路和奇妙故事都分享给大…

vue中如何动态的增减组件的类名(class)

在 Vue.js 2 中&#xff0c;你可以通过计算属性或直接在模板中使用 v-bind:class 来动态地改变组件的类名。下面是一个简单的示例&#xff0c;说明如何在某个条件被复核后为组件添加一个 selected 类&#xff08;此处为组件添加一个默认的类&#xff08;例如 radio&#xff09;…

Vue3 基础概念与环境搭建

一、Vue3 简介 Vue3 是 Vue.js 的最新主要版本&#xff0c;于 2020 年 9 月正式发布。它在性能、可维护性和开发体验方面都有了显著的改进。相比 Vue2&#xff0c;Vue3 的主要特点包括&#xff1a; 更高效的响应式系统&#xff1a;使用 Proxy替代了 Object.defineProperty&…

华为昇腾920b服务器部署DeepSeek翻车现场

最近到祸一台HUAWEI Kunpeng 920 5250&#xff0c;先看看配置。之前是部署的讯飞大模型&#xff0c;发现资源利用率太低了。把5台减少到3台&#xff0c;就出了他 硬件配置信息 基本硬件信息 按照惯例先来看看配置。一共3块盘&#xff0c;500G的系统盘&#xff0c; 2块3T固态…

Python的那些事第二十三篇:Express(Node.js)与 Python:一场跨语言的浪漫邂逅

摘要 在当今的编程世界里,Node.js 和 Python 像是两个性格迥异的超级英雄,一个以速度和灵活性著称,另一个则以强大和优雅闻名。本文将探讨如何通过 Express 框架将 Node.js 和 Python 结合起来,打造出一个高效、有趣的 Web 应用。我们将通过一系列幽默风趣的实例和表格,展…

Word中接入大模型教程

前言 为什么要在word中接入大模型呢&#xff1f; 个人觉得最大的意义就是不用来回切换与复制粘贴了吧。 今天分享一下昨天实践的在word中接入大模型的教程。 在word中接入大模型最简单的方式就是使用vba。 vba代码要做的事&#xff0c;拆分一下就是&#xff1a; 获取用户…

open3d绘制平面

在Open3D中绘制平面通常涉及到创建一个平面模型并将其可视化。Open3D是一个开源库,主要用于3D数据的处理和可视化,但它主要用于3D数据的处理,并不直接支持绘制2D平面。如果你想在Open3D中“绘制”一个平面,你可以通过以下几种方法来实现类似的效果: 方法1:使用o3d.geome…

DeepSeek R1 与 OpenAI O1:机器学习模型的巅峰对决

我的个人主页 我的专栏&#xff1a;人工智能领域、java-数据结构、Javase、C语言&#xff0c;希望能帮助到大家&#xff01;&#xff01;&#xff01;点赞&#x1f44d;收藏❤ 一、引言 在机器学习的广袤天地中&#xff0c;大型语言模型&#xff08;LLM&#xff09;无疑是最…

WebGPU顶点插槽进阶优化指南:释放GPU渲染性能

本文基于WebGPU官方规范与实践经验&#xff0c;深入探讨顶点缓冲区的性能优化策略&#xff0c;涵盖数据布局、资源管理、渲染流程等多个维度&#xff0c;并附详细代码注释与性能对比分析。 一、数据布局优化&#xff1a;降低内存与带宽压力 1. 内存对齐策略 GPU对内存访问有严…