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

【设计模式-3.7】结构型——组合模式

说明:本文介绍结构型设计模式之一的组合模式

定义

组合模式(Composite Pattern)又叫作整体-部分(Part-Whole)模式,它的宗旨是通过将单个对象(叶子节点)和组合对象(树枝节点)用相同的接口进行表示,使得客户对单个对象和组合对象的使用具有一致性,属于结构型设计模式。(引自《设计模式就该这样学》P263)

文件系统

以文件系统为例,如下,是服务器上某个文件夹的文件结构,该文件夹下既有文件夹,也有文件

在这里插入图片描述

如果我要构建这样一个文件夹-文件系统,代码应该是这样写的,如下:

(图片文件,ImageFile)

/*** 图片文件*/
public class ImageFile {/*** 图片名称*/private String name;public ImageFile(String name) {this.name = name;}/*** 展示*/public void show(int space) {for (int i = 0; i < space; i++) {System.out.print(" ");}System.out.println(name);}
}

(电影文件,MovieFile)

/*** 电影文件*/
public class MovieFile {/*** 电影名称*/private String name;public MovieFile(String name) {this.name = name;}/*** 展示*/public void show(int space) {for (int i = 0; i < space; i++) {System.out.print(" ");}System.out.println(name);}
}

(文件夹,Folder,定义多个文件集合,并开放对应的增加方法)

import java.util.ArrayList;/*** 文件夹*/
public class Folder {/*** 文件夹名称*/private String name;/*** 文件夹下的文件夹*/private ArrayList<Folder> folders = new ArrayList<>();/*** 文件夹下的图片文件*/private ArrayList<ImageFile> imageFiles = new ArrayList<>();/*** 文件夹下的视频文件*/private ArrayList<MovieFile> movieFiles = new ArrayList<>();public Folder(String name) {this.name = name;}/*** 添加文件夹*/public void addFolder(Folder folder) {folders.add(folder);}/*** 添加图片文件*/public void addImageFile(ImageFile imageFile) {imageFiles.add(imageFile);}/*** 添加电影文件*/public void addMoveFile(MovieFile movieFile) {movieFiles.add(movieFile);}/*** 展示*/public void show(int space) {// 打印文件夹space++;for (int i = 0; i < space; i++) {System.out.print(" ");}System.out.println(name);for (Folder folder : folders) {folder.show(space);}// 打印图片文件space++;for (ImageFile imageFile : imageFiles) {imageFile.show(space);}// 打印电影文件space++;for (MovieFile movieFile : movieFiles) {movieFile.show(space);}}
}

(客户端使用,Client)

public class Client {public static void main(String[] args) {// 顶级文件夹Folder folder = new Folder("folder");// 二级文件夹Folder images = new Folder("images");Folder movies = new Folder("movies");// 三级目录下的文件ImageFile boy = new ImageFile("boy.png");ImageFile girl = new ImageFile("girl.png");images.addImageFile(boy);images.addImageFile(girl);// 三级文件夹Folder director1 = new Folder("heizeming");Folder director2 = new Folder("xiaolinzhengshu");// 四级目录MovieFile movieFile1 = new MovieFile("luoshengmen.mp4");MovieFile movieFile2 = new MovieFile("qiwushi.mp4");director1.addMoveFile(movieFile1);director1.addMoveFile(movieFile2);MovieFile movieFile3 = new MovieFile("duomingjian.mp4");MovieFile movieFile4 = new MovieFile("qiefu.mp4");director2.addMoveFile(movieFile3);director2.addMoveFile(movieFile4);movies.addFolder(director1);movies.addFolder(director2);// 文件夹添加到顶级文件夹folder.addFolder(images);folder.addFolder(movies);folder.show(0);}
}

运行,hora!(看!),能实现目的

在这里插入图片描述

但这里存在问题,文件系统,简单来说只有文件夹和文件两个实体对象,代码中的图片文件、电影文件可以抽象为文件(File),这是整体与个体的场景。

组合模式

使用组合模式改进上述代码,如下:

(抽象节点类,Node)

/*** 抽象节点*/
public abstract class Node {/*** 节点名称*/protected String name;public Node(String name) {this.name = name;}/*** 添加节点*/protected abstract void add(Node node);/*** 展示*/protected void show(int space) {for (int i = 0; i < space; i++) {System.out.print(" ");}System.out.println(name);}/*** 重载方法,使用的时候就不用给参数了*/protected void show() {show(0);}
}

(文件夹,Folder)

import java.util.ArrayList;/*** 文件夹*/
public class Folder extends Node {/*** 文件夹下的子节点*/private ArrayList<Node> childrenNodes = new ArrayList<>();/*** 调用父类的构造方法*/public Folder(String name) {super(name);}@Overrideprotected void add(Node node) {childrenNodes.add(node);}@Overrideprotected void show(int space) {super.show(space);space++;for (Node node : childrenNodes) {node.show(space);}}
}

(文件,File)

/*** 文件*/
public class File extends Node {/*** 调用父类的构造方法*/public File(String name) {super(name);}@Overrideprotected void add(Node node) {System.out.println("不能添加子节点");}@Overrideprotected void show(int space) {super.show(space);}
}

(客户端使用,Client)

public class Client {public static void main(String[] args) {// 顶级文件夹Folder folder = new Folder("folder");// 二级文件夹Folder images = new Folder("images");Folder movies = new Folder("movies");// 三级目录下的文件File boy = new File("boy.png");File girl = new File("girl.png");images.add(boy);images.add(girl);// 三级文件夹Folder director1 = new Folder("heizeming");Folder director2 = new Folder("xiaolinzhengshu");// 四级目录File movieFile1 = new File("luoshengmen.mp4");File movieFile2 = new File("qiwushi.mp4");director1.add(movieFile1);director1.add(movieFile2);File movieFile3 = new File("duomingjian.mp4");File movieFile4 = new File("qiefu.mp4");director2.add(movieFile3);director2.add(movieFile4);movies.add(director1);movies.add(director2);// 文件夹添加到顶级文件夹folder.add(images);folder.add(movies);folder.show();}
}

执行如下,也实现了目的

在这里插入图片描述

这么看下来,文件系统场景使用组合模式实现是很不错的,代码少了很多,也削减了文件夹类中的职责(可以对比下Folder类前后的代码)

使用场景

在《设计模式就该这样学》(P229)这本书中,提到状态模式适用于以下场景:

(1)希望客户端可以忽略组合对象与单个对象的差异;

(2)对象层次具备整体和部分,呈树形结构;

除了文件系统、企业组织架构场景,我还没想到其他使用场景;

总结

本文介绍了结构型设计模式中的组合模式,参考《设计模式就该这样学》、《秒懂设计模式》、《设计模式的艺术》(第一版)这三本书,其中的例子来自《秒懂设计模式》。

http://www.lryc.cn/news/2401621.html

相关文章:

  • Unity Mac 笔记本操作入门
  • 实时数据仓库是什么?数据仓库设计怎么做?
  • Linux(12)——基础IO(下)
  • WPF可拖拽ListView
  • rocketmq索引
  • [蓝桥杯]倍数问题
  • 定时任务的 cron 表达式
  • 【MySQL】 约束
  • MySQL 的 redo log 和 binlog 区别?
  • 前端vue打开多个窗口,关闭窗口后才继续执行后续逻辑
  • 「深度拆解」Spring Boot如何用DeepSeek重构MCP通信层?从线程模型到分布式推理的架构进化
  • 如何避免在前端项目中出现重复的第三方依赖包?
  • Java开发中复用公共SQL的方法
  • 【西门子杯工业嵌入式-2-点亮一颗LED】
  • 代码随想录算法训练营第60期第五十五天打卡
  • 重磅更新! 基于Gemini 2.5 Pro打造的AI智能体PlantUML-X上线!
  • [5-02-04].第01节:Jmeter环境搭建:
  • AI智能推荐实战之RunnableParallel并行链
  • windows server2019 不成功的部署docker经历
  • Gemini开源项目DeepResearch:基于LangGraph的智能研究代理技术原理与实现
  • React状态管理Context API + useReducer
  • 【无标题】路径着色问题的革命性重构:拓扑色动力学模型下的超越与升华
  • Doris Catalog 联邦分析查询性能优化:从排查到优化的完整指南
  • 01 Deep learning神经网络的编程基础 二分类--吴恩达
  • 视频自动化分割方案:支持按时间与段数拆分
  • Open SSL 3.0相关知识以及源码流程分析
  • 股指期货合约价值怎么算?
  • 【QT】使用QT帮助手册找控件样式
  • 计算机网络(5)——数据链路层
  • VuePress完美整合Toast消息提示