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

(十 一)趣学设计模式 之 组合模式!

在这里插入图片描述

目录

    • 一、 啥是组合模式?
    • 二、 为什么要用组合模式?
    • 三、 组合模式的实现方式
    • 四、 组合模式的优缺点
    • 五、 组合模式的应用场景
    • 六、 总结

🌟我的其他文章也讲解的比较有趣😁,如果喜欢博主的讲解方式,可以多多支持一下,感谢🤗!
🌟了解外观模式请看: (十)趣学设计模式 之 外观模式!

这篇文章带你详细认识一下设计模式中的外观模式

一、 啥是组合模式?

想象一下,你正在整理你的电脑文件 📁。 你有文件,也有文件夹。 文件夹里可以包含文件,也可以包含其他文件夹。 你可以对文件和文件夹进行统一的操作,比如复制、删除、移动等等 ✂️。
在这里插入图片描述

组合模式,就是将对象组合成树形结构以表示“整体-部分”的层次结构! 组合模式使得用户对单个对象和组合对象的使用具有一致性 🌳。

简单来说,就是把一组相似的对象组织成树状结构,方便统一管理和操作! 🌲

  • 你想表示“整体-部分”的层次结构: 就像文件夹包含文件和文件夹 📁!
  • 你想让客户端可以统一地处理单个对象和组合对象: 就像你可以对文件和文件夹进行统一的复制、删除操作 ✂️!
  • 你想忽略单个对象和组合对象之间的差异: 就像你不想区分文件和文件夹,都把它们当作“文件系统对象”来处理 📁📄!

二、 为什么要用组合模式?

用组合模式,好处多多 👍:

  • 表示层次结构: 可以清晰地表示“整体-部分”的层次结构 🌳!
  • 统一操作: 可以统一地操作单个对象和组合对象 ✂️!
  • 简化客户端代码: 客户端不需要区分单个对象和组合对象,代码更简洁 🧹!
  • 扩展性好: 可以方便地添加新的叶子节点和组合节点 ➕!

三、 组合模式的实现方式

组合模式主要包含以下几个角色:

  • Component(组件): 定义组合中叶子节点和组合节点的通用接口。 📁📄 (比如:文件系统对象)
  • Leaf(叶子节点): 表示组合中的叶子节点,没有子节点。 📄 (比如:文件)
  • Composite(组合节点): 表示组合中的组合节点,可以包含叶子节点和其他组合节点。 📁 (比如:文件夹)

代码示例:

import java.util.ArrayList;
import java.util.List;// 组件:文件系统对象
public abstract class FileSystemObject {protected String name; // 名称protected int level = 0; // 层次级别public FileSystemObject(String name) {this.name = name;}public abstract void display(); // 显示// 设置层次级别public void setLevel(int level) {this.level = level;}// 获取缩进字符串protected String getIndentString() {StringBuilder sb = new StringBuilder();for (int i = 0; i < level; i++) {sb.append("  "); // 两个空格缩进}return sb.toString();}
}// 叶子节点:文件
public class File extends FileSystemObject {public File(String name) {super(name);}@Overridepublic void display() {System.out.println(getIndentString() + "文件:" + name);}
}// 组合节点:文件夹
public class Directory extends FileSystemObject {private List<FileSystemObject> children = new ArrayList<>(); // 子节点public Directory(String name) {super(name);}public void add(FileSystemObject fileSystemObject) {children.add(fileSystemObject);}public void remove(FileSystemObject fileSystemObject) {children.remove(fileSystemObject);}@Overridepublic void display() {System.out.println(getIndentString() + "文件夹:" + name);for (FileSystemObject child : children) {child.setLevel(this.level + 1); // 设置子节点的层次级别child.display();}}
}// 客户端
public class Client {public static void main(String[] args) {Directory root = new Directory("根目录"); // 创建根目录Directory dir1 = new Directory("目录1"); // 创建目录1File file1 = new File("文件1.txt"); // 创建文件1File file2 = new File("文件2.txt"); // 创建文件2root.add(dir1); // 将目录1添加到根目录root.add(file1); // 将文件1添加到根目录dir1.add(file2); // 将文件2添加到目录1root.display(); // 显示根目录}
}

分析:

  • FileSystemObject 是组件,定义了文件系统对象的通用接口。
  • File 是叶子节点,表示文件。
  • Directory 是组合节点,表示文件夹,可以包含文件和文件夹。

输出结果:

文件夹:根目录文件夹:目录1文件:文件2.txt文件:文件1.txt

四、 组合模式的优缺点

优点:

  • 表示层次结构 🌳!
  • 统一操作 ✂️!
  • 简化客户端代码 🧹!
  • 扩展性好 ➕!

缺点:

  • 设计较复杂,客户端需要花更多时间理清类之间的层次关系 😫!
  • 不容易限制组合中的组件类型,可能会导致运行时错误 💥!

五、 组合模式的应用场景

  • 你想表示“整体-部分”的层次结构: 就像文件系统 📁!
  • 你想让客户端可以统一地处理单个对象和组合对象: 就像你可以对文件和文件夹进行统一的复制、删除操作 ✂️!
  • GUI组件: GUI组件通常是树形结构,比如窗口包含按钮、文本框等组件。
  • 组织结构: 公司组织结构可以表示为树形结构,部门包含员工和子部门。

六、 总结

  • 组合模式就像把一组相似的对象组织成树状结构,方便统一管理和操作! 🌲
  • 主要包含组件、叶子节点和组合节点三个角色! 🎭
  • 优点是表示层次结构、统一操作、简化客户端代码、扩展性好! 👍
  • 缺点是设计较复杂、不容易限制组合中的组件类型! 👎
  • 适用于需要表示“整体-部分”的层次结构,或者需要统一地处理单个对象和组合对象的场景! 🎯

希望这篇文章能让你彻底理解组合模式! 💯 祝你学习愉快! 😄
看完请看:(十 二)趣学设计模式 之 享元模式!

相关文章:

(十 一)趣学设计模式 之 组合模式!

目录 一、 啥是组合模式&#xff1f;二、 为什么要用组合模式&#xff1f;三、 组合模式的实现方式四、 组合模式的优缺点五、 组合模式的应用场景六、 总结 &#x1f31f;我的其他文章也讲解的比较有趣&#x1f601;&#xff0c;如果喜欢博主的讲解方式&#xff0c;可以多多支…...

安全模块设计:token服务、校验注解(开启token校验、开启签名校验、允许处理API日志)、获取当前用户信息的辅助类

文章目录 引言pom.xmlI 校验注解ApiValidationII token服务TokenService获取当前用户信息的辅助类III 域登录接口响应数据登陆用户信息引言 pom.xml <?xml version="1.0" encoding="UTF-8"?> <project xmlns="http://maven.apache.org/PO…...

Python学习第十八天之深度学习之Tensorboard

Tensorboard 1.TensorBoard详解2.安装3.使用4.图像数据格式的一些理解 后续会陆续在词博客上更新Tensorboard相关知识 1.TensorBoard详解 TensorBoard是一个可视化的模块&#xff0c;该模块功能强大&#xff0c;可用于深度学习网络模型训练查看模型结构和训练效果&#xff08;…...

Redis安装及其AnotherRedisDesktopManagera安装使用

一、Redis安装 1. 下载Redis安装包 通过网盘分享的文件&#xff1a;Redis 链接: https://pan.baidu.com/s/1elAT8mk3EIoYQQ3WoVVoNg?pwd7yrz 提取码: 7yrz 2. 解压Redis安装包 下载完成后&#xff0c;将Redis安装包解压到一个指定的目录&#xff0c;例如&#xff1a;C:\Re…...

C# dll文件的反编译获取源码

目录 前言操作流程结论 前言 上一篇文章介绍了将C# cs类文件加密为dll文件&#xff0c;在此给大家写一篇关于反编译dll文件的文章。 操作流程 首先&#xff0c;我们需要准备一个C#反编译工具&#xff0c;我这里用的是免费的软件JetBrains dotPeek&#xff0c;类似的有很多&am…...

大语言模型学习--LangChain

LangChain基本概念 ReAct学习资料 https://zhuanlan.zhihu.com/p/660951271 LangChain官网地址 Introduction | &#x1f99c;️&#x1f517; LangChain LangChain是一个基于语言模型开发应用程序的框架。它可以实现以下应用程序&#xff1a; 数据感知&#xff1a;将语言模型…...

Spark内存迭代计算

一、宽窄依赖 窄依赖&#xff1a;父RDD的一个分区数据全部发往子RDD的一个分区 宽依赖&#xff1a;父RDD的一个分区数据发往子RDD的多个分区&#xff0c;也称为shuffle 二、Spark是如何进行内存计算的&#xff1f;DAG的作用&#xff1f;Stage阶段划分的作用&#xff1f; &a…...

Python之参数星号(*)使用笔记

背景 在学习python时发现方法调用和方法定义会经常发现有带星号的标记&#xff0c;为了弄明白是怎么使用的。特此做个笔记。 一、参数符号对比速查表 符号类使用场景作用描述示例无符号函数定义/调用普通位置参数或关键字参数.def func(a, b)*函数定义收集多余位置参数为元组…...

一文掌握 Scrapy 框架的详细使用,包括实战案例

更多内容请见: 爬虫和逆向教程-专栏介绍和目录 文章目录 1. Scrapy 简介2. Scrapy 的核心组件3. 安装 Scrapy4. 创建 Scrapy 项目4.1 创建项目4.2 创建 Spider5. 编写 Spider5.1 定义 Item5.2 编写 Spider 逻辑6. 运行 Scrapy 爬虫6.1 运行爬虫6.2 保存爬取数据7. Scrapy 的高…...

【Mac】git使用再学习

目录 前言 如何使用github建立自己的代码库 第一步&#xff1a;建立本地git与远程github的联系 生成密钥 将密钥加入github 第二步&#xff1a;创建github仓库并clone到本地 第三步&#xff1a;上传文件 常见的git命令 git commit git branch git merge/git rebase …...

【MySQL篇】数据库基础

目录 1&#xff0c;什么是数据库&#xff1f; 2&#xff0c;主流数据库 3&#xff0c;MySQL介绍 1&#xff0c;MySQL架构 2&#xff0c;SQL分类 3&#xff0c;MySQL存储引擎 1&#xff0c;什么是数据库&#xff1f; 数据库&#xff08;Database&#xff0c;简称DB&#xf…...

SpringBoot项目注入 traceId 来追踪整个请求的日志链路

SpringBoot项目注入 traceId 来追踪整个请求的日志链路&#xff0c;有了 traceId&#xff0c; 我们在排查问题的时候&#xff0c;可以迅速根据 traceId 查找到相关请求的日志&#xff0c;特别是在生产环境的时候&#xff0c;用户可能只提供一个错误截图&#xff0c;我们作为开发…...

【Block总结】SAFMN,空间自适应调制与局部特征增强的协同设计|即插即用

论文信息 标题&#xff1a;Spatially-Adaptive Feature Modulation for Efficient Image Super-Resolution论文链接&#xff1a;https://arxiv.org/pdf/2302.13800代码与模型&#xff1a;https://github.com/sunny2109/SAFMN 创新点 空间自适应特征调制&#xff08;SAFM&…...

Python爬虫:一文掌握PyQuery模块

文章目录 1. PyQuery 简介2. PyQuery 的安装2.1 安装 PyQuery2.2 安装依赖库3. PyQuery 的基本使用3.1 初始化 PyQuery 对象3.2 选择元素3.3 获取元素内容3.4 遍历元素4. PyQuery 的高级用法4.1 过滤元素4.2 查找子元素4.3 获取属性值4.4 修改元素4.5 添加和删除元素4.6 遍历文…...

LearnOpenGL之Shader编程用算法绘画

———————————————————— 前序 ——————————————————— AndroidLearnOpenGL是本博主自己实现的LearnOpenGL练习集合&#xff1a; Github地址&#xff1a;GitHub - wangyongyao1989/AndroidLearnOpenGL: OpenGL基础及运用 系列文章&#xff…...

如何使用Spring Boot框架整合Redis:超详细案例教程

目录 # 为什么选择Spring Boot与Redis整合&#xff1f; 1. 更新 pom.xml 2. 配置application.yml 3. 创建 Redis 配置类 4. Redis 操作类 5. 创建控制器 6. 启动应用程序 7. 测试 # 为什么选择Spring Boot与Redis整合&#xff1f; 将Spring Boot与Redis整合可以充分利…...

算法--贪心

贪心 原理经典例题[860. 柠檬水找零](https://leetcode.cn/problems/lemonade-change/description/)[2208. 将数组和减半的最少操作次数](https://leetcode.cn/problems/minimum-operations-to-halve-array-sum/description/)[179. 最大数](https://leetcode.cn/problems/large…...

线程控制(创建、终止、等待、分离)

目录 1.前言 2.创建线程 pthread_create函数 3.线程终止 pthread_exit函数 pthread_cancel函数 4.线程等待 5.线程分离 1.前言 在Linux系统中&#xff0c;并不存在真正的线程&#xff0c;只有轻量级进程。所以&#xff0c;Linux系统只提供了操作轻量级进程的系统调用…...

【备份】php项目处理跨域请求踩坑

这都是老生常谈的东西了。我还在踩坑&#xff0c;记录一下。 我在项目入口明明写了如下代码&#xff1a; // 处理预检请求 (OPTIONS) if ($_SERVER[REQUEST_METHOD] OPTIONS) {header("Access-Control-Allow-Origin: https://xxx.vip");header("Access-Cont…...

目标检测YOLO实战应用案例100讲-面向无人机图像的小目标检测

目录 知识储备 YOLO v8无人机拍摄视角小目标检测 数据集结构 环境部署说明 安装依赖 模型训练权重和指标可视化展示 训练 YOLOv8 PyQt5 GUI 开发 主窗口代码 main_window.py 使用说明 无人机目标跟踪 一、目标跟踪的基本原理 二、常用的目标跟踪算法 基于YOLOv…...

web vue 项目 Docker化部署

Web 项目 Docker 化部署详细教程 目录 Web 项目 Docker 化部署概述Dockerfile 详解 构建阶段生产阶段 构建和运行 Docker 镜像 1. Web 项目 Docker 化部署概述 Docker 化部署的主要步骤分为以下几个阶段&#xff1a; 构建阶段&#xff08;Build Stage&#xff09;&#xff1a…...

ES6从入门到精通:前言

ES6简介 ES6&#xff08;ECMAScript 2015&#xff09;是JavaScript语言的重大更新&#xff0c;引入了许多新特性&#xff0c;包括语法糖、新数据类型、模块化支持等&#xff0c;显著提升了开发效率和代码可维护性。 核心知识点概览 变量声明 let 和 const 取代 var&#xf…...

Zustand 状态管理库:极简而强大的解决方案

Zustand 是一个轻量级、快速和可扩展的状态管理库&#xff0c;特别适合 React 应用。它以简洁的 API 和高效的性能解决了 Redux 等状态管理方案中的繁琐问题。 核心优势对比 基本使用指南 1. 创建 Store // store.js import create from zustandconst useStore create((set)…...

Java如何权衡是使用无序的数组还是有序的数组

在 Java 中,选择有序数组还是无序数组取决于具体场景的性能需求与操作特点。以下是关键权衡因素及决策指南: ⚖️ 核心权衡维度 维度有序数组无序数组查询性能二分查找 O(log n) ✅线性扫描 O(n) ❌插入/删除需移位维护顺序 O(n) ❌直接操作尾部 O(1) ✅内存开销与无序数组相…...

【SpringBoot】100、SpringBoot中使用自定义注解+AOP实现参数自动解密

在实际项目中,用户注册、登录、修改密码等操作,都涉及到参数传输安全问题。所以我们需要在前端对账户、密码等敏感信息加密传输,在后端接收到数据后能自动解密。 1、引入依赖 <dependency><groupId>org.springframework.boot</groupId><artifactId...

基于Flask实现的医疗保险欺诈识别监测模型

基于Flask实现的医疗保险欺诈识别监测模型 项目截图 项目简介 社会医疗保险是国家通过立法形式强制实施&#xff0c;由雇主和个人按一定比例缴纳保险费&#xff0c;建立社会医疗保险基金&#xff0c;支付雇员医疗费用的一种医疗保险制度&#xff0c; 它是促进社会文明和进步的…...

关于iview组件中使用 table , 绑定序号分页后序号从1开始的解决方案

问题描述&#xff1a;iview使用table 中type: "index",分页之后 &#xff0c;索引还是从1开始&#xff0c;试过绑定后台返回数据的id, 这种方法可行&#xff0c;就是后台返回数据的每个页面id都不完全是按照从1开始的升序&#xff0c;因此百度了下&#xff0c;找到了…...

【解密LSTM、GRU如何解决传统RNN梯度消失问题】

解密LSTM与GRU&#xff1a;如何让RNN变得更聪明&#xff1f; 在深度学习的世界里&#xff0c;循环神经网络&#xff08;RNN&#xff09;以其卓越的序列数据处理能力广泛应用于自然语言处理、时间序列预测等领域。然而&#xff0c;传统RNN存在的一个严重问题——梯度消失&#…...

MMaDA: Multimodal Large Diffusion Language Models

CODE &#xff1a; https://github.com/Gen-Verse/MMaDA Abstract 我们介绍了一种新型的多模态扩散基础模型MMaDA&#xff0c;它被设计用于在文本推理、多模态理解和文本到图像生成等不同领域实现卓越的性能。该方法的特点是三个关键创新:(i) MMaDA采用统一的扩散架构&#xf…...

质量体系的重要

质量体系是为确保产品、服务或过程质量满足规定要求&#xff0c;由相互关联的要素构成的有机整体。其核心内容可归纳为以下五个方面&#xff1a; &#x1f3db;️ 一、组织架构与职责 质量体系明确组织内各部门、岗位的职责与权限&#xff0c;形成层级清晰的管理网络&#xf…...