组合模式:如何设计实现支持递归遍历的文件系统目录树结构?

        组合模式跟我们之前讲的面向对象设计中的“组合关系(通过组合来组装两个类)”,完全是两码事。这里讲的“组合模式”,主要是用来处理树形结构数据。这里的“数据”,你可以简单理解为一组对象集合,待会我们会详细讲解。

正因为其应用场景的特殊性,数据必须能表示成树形结构,这也导致了这种模式在实际的项目开发中并不那么常用。

将一组对象组织(Compose)成树形结构,以表示一种“部分-整体”的层次结构。组合让客户端(在很多设计模式书籍中,“客户端”代指代码的使用者。)可以统一单个对象和组合对象的处理逻辑。

假设我们有这样一个需求:设计一个类来表示文件系统中的目录,能方便地实现下面这些功能:

  • 动态地添加、删除某个目录下的子目录或文件;
  • 统计指定目录下的文件个数;
  • 统计指定目录下的文件总大小。
public class FileSystemNode {private String path;private boolean isFile;private List<FileSystemNode> subNodes = new ArrayList<>();public FileSystemNode(String path, boolean isFile) {this.path = path;this.isFile = isFile;}public int countNumOfFiles() {// TODO:...if (isFile) {return 1;}int numOfFiles = 0;for (FileSystemNode fileOrDir : subNodes) {numOfFiles += fileOrDir.countNumOfFiles();}return numOfFiles;}public long countSizeOfFiles() {// TODO:...if (isFile) {File file = new File(path);if (!file.exists()) return 0;return file.length();}long sizeofFiles = 0;for (FileSystemNode fileOrDir : subNodes) {sizeofFiles += fileOrDir.countSizeOfFiles();}return sizeofFiles;}public String getPath() {return path;}public void addSubNode(FileSystemNode fileOrDir) {subNodes.add(fileOrDir);}public void removeSubNode(FileSystemNode fileOrDir) {int size = subNodes.size();int i = 0;for (; i < size; ++i) {if (subNodes.get(i).getPath().equalsIgnoreCase(fileOrDir.getPath())) {break;}}if (i < size) {subNodes.remove(i);}}
}

单纯从功能实现角度来说,上面的代码没有问题,已经实现了我们想要的功能。但是,如果我们开发的是一个大型系统,从扩展性(文件或目录可能会对应不同的操作)、业务建模(文件和目录从业务上是两个概念)、代码的可读性(文件和目录区分对待更加符合人们对业务的认知)的角度来说,我们最好对文件和目录进行区分设计,定义为File和Directory两个类

public abstract class FileSystemNode {protected String path;public FileSystemNode(String path) {this.path = path;}public abstract int countNumOfFiles();public abstract long countSizeOfFiles();public String getPath() {return path;}
}public class File extends FileSystemNode {public File(String path) {super(path);}@Overridepublic int countNumOfFiles() {return 1;}@Overridepublic long countSizeOfFiles() {java.io.File file = new java.io.File(path);if (!file.exists()) return 0;return file.length();}
}public class Directory extends FileSystemNode {private List<FileSystemNode> subNodes = new ArrayList<>();public Directory(String path) {super(path);}@Overridepublic int countNumOfFiles() {int numOfFiles = 0;for (FileSystemNode fileOrDir : subNodes) {numOfFiles += fileOrDir.countNumOfFiles();}return numOfFiles;}@Overridepublic long countSizeOfFiles() {long sizeofFiles = 0;for (FileSystemNode fileOrDir : subNodes) {sizeofFiles += fileOrDir.countSizeOfFiles();}return sizeofFiles;}public void addSubNode(FileSystemNode fileOrDir) {subNodes.add(fileOrDir);}public void removeSubNode(FileSystemNode fileOrDir) {int size = subNodes.size();int i = 0;for (; i < size; ++i) {if (subNodes.get(i).getPath().equalsIgnoreCase(fileOrDir.getPath())) {break;}}if (i < size) {subNodes.remove(i);}}
}

文件和目录类都设计好了,我们来看,如何用它们来表示一个文件系统中的目录树结构。具体的代码示例如下所示:

public class Demo {public static void main(String[] args) {/*** /* /wz/* /wz/a.txt* /wz/b.txt* /wz/movies/* /wz/movies/c.avi* /xzg/* /xzg/docs/* /xzg/docs/d.txt*/Directory fileSystemTree = new Directory("/");Directory node_wz = new Directory("/wz/");Directory node_xzg = new Directory("/xzg/");fileSystemTree.addSubNode(node_wz);fileSystemTree.addSubNode(node_xzg);File node_wz_a = new File("/wz/a.txt");File node_wz_b = new File("/wz/b.txt");Directory node_wz_movies = new Directory("/wz/movies/");node_wz.addSubNode(node_wz_a);node_wz.addSubNode(node_wz_b);node_wz.addSubNode(node_wz_movies);File node_wz_movies_c = new File("/wz/movies/c.avi");node_wz_movies.addSubNode(node_wz_movies_c);Directory node_xzg_docs = new Directory("/xzg/docs/");node_xzg.addSubNode(node_xzg_docs);File node_xzg_docs_d = new File("/xzg/docs/d.txt");node_xzg_docs.addSubNode(node_xzg_docs_d);System.out.println("/ files num:" + fileSystemTree.countNumOfFiles());System.out.println("/wz/ files num:" + node_wz.countNumOfFiles());}
}

我们对照着这个例子,再重新看一下组合模式的定义:“将一组对象(文件和目录)组织成树形结构,以表示一种‘部分-整体’的层次结构(目录与子目录的嵌套结构)。组合模式让客户端可以统一单个对象(文件)和组合对象(目录)的处理逻辑(递归遍历)。”

上述:一个文件表示单个对象,一个目录表示组合对象。他们的处理逻辑都是

countNumOfFiles 和 countSizeOfFiles

实际上,刚才讲的这种组合模式的设计思路,与其说是一种设计模式,倒不如说是对业务场景的一种数据结构和算法的抽象。其中,数据可以表示成树这种数据结构,业务需求可以通过在树上的递归遍历算法来实现。

组合模式的应用场景举例

刚刚我们讲了文件系统的例子,对于组合模式,我这里再举一个例子。搞懂了这两个例子,你基本上就算掌握了组合模式。在实际的项目中,遇到类似的可以表示成树形结构的业务场景,你只要“照葫芦画瓢”去设计就可以了。

假设我们在开发一个OA系统(办公自动化系统)。公司的组织结构包含部门和员工两种数据类型。其中,部门又可以包含子部门员工。在数据库中的表结构如下所示:

 

 

我们希望在内存中构建整个公司的人员架构图(部门、子部门、员工的隶属关系),并且提供接口计算出部门的薪资成本(隶属于这个部门的所有员工的薪资和)。

部门包含子部门和员工,这是一种嵌套结构,可以表示成树这种数据结构。计算每个部门的薪资开支这样一个需求,也可以通过在树上的遍历算法来实现。所以,从这个角度来看,这个应用场景可以使用组合模式来设计和实现。

这个例子的代码结构跟上一个例子的很相似,代码实现我直接贴在了下面,你可以对比着看一下。其中,HumanResource是部门类(Department)和员工类(Employee)抽象出来的父类,为的是能统一薪资的处理逻辑。Demo中的代码负责从数据库中读取数据并在内存中构建组织架构图。

public abstract class HumanResource {protected long id;protected double salary;public HumanResource(long id) {this.id = id;}public long getId() {return id;}public abstract double calculateSalary();
}public class Employee extends HumanResource {public Employee(long id, double salary) {super(id);this.salary = salary;}@Overridepublic double calculateSalary() {return salary;}
}public class Department extends HumanResource {private List<HumanResource> subNodes = new ArrayList<>();public Department(long id) {super(id);}@Overridepublic double calculateSalary() {double totalSalary = 0;for (HumanResource hr : subNodes) {totalSalary += hr.calculateSalary();}this.salary = totalSalary;return totalSalary;}public void addSubNode(HumanResource hr) {subNodes.add(hr);}
}// 构建组织架构的代码
public class Demo {private static final long ORGANIZATION_ROOT_ID = 1001;private DepartmentRepo departmentRepo; // 依赖注入private EmployeeRepo employeeRepo; // 依赖注入public void buildOrganization() {Department rootDepartment = new Department(ORGANIZATION_ROOT_ID);buildOrganization(rootDepartment);}private void buildOrganization(Department department) {List<Long> subDepartmentIds = departmentRepo.getSubDepartmentIds(department.getId());for (Long subDepartmentId : subDepartmentIds) {Department subDepartment = new Department(subDepartmentId);department.addSubNode(subDepartment);buildOrganization(subDepartment);}List<Long> employeeIds = employeeRepo.getDepartmentEmployeeIds(department.getId());for (Long employeeId : employeeIds) {double salary = employeeRepo.getEmployeeSalary(employeeId);department.addSubNode(new Employee(employeeId, salary));}}
}

我们再拿组合模式的定义跟这个例子对照一下:“将一组对象(员工和部门)组织成树形结构,以表示一种‘部分-整体’的层次结构(部门与子部门的嵌套结构)。组合模式让客户端可以统一单个对象(员工)和组合对象(部门)的处理逻辑(递归遍历)。”

此时这里的个体指的是:员工,组织指的是部门。

主要符合这种树形关系的都可以使用组合模式

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

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

相关文章

JVM04-优化JVM内存分配以及内存持续上升问题和CPU过高问题排查

1-JVM内存分配 1.1-JVM内存分配性能问题 JVM内存分配不合理最直接的表现就是频繁的GC&#xff0c;这会导致上下文切换等性能问题&#xff0c;从而降低系统的吞吐量、增加系统的响应时间。因此&#xff0c;如果你在线上环境或性能测试时&#xff0c;发现频繁的GC&#xff0c;且…

Spring异常处理器

文章目录 1. 异常分析2. 异常处理器2.1 异常处理器核心2.2 异常处理顺序 3. 自定义异常 1. 异常分析 问题:   程序允许不免的在各层都可能会产生异常&#xff0c;我们该如何处理这些异常? 如果只是在方法里面单独使用 try… catch… 语句去一个一个的进行捕捉处理的话&#x…

MAC M1上docker rocketmq简单环境搭建和代码

工作了这么多年&#xff0c;rocketmq还没有用过&#xff0c;由于现在的工作中涉及到了&#xff0c;周六吃完午饭就开始搞&#xff0c;结果到现在3点钟才把环境弄好&#xff0c;测试代码搞起。 整个流程分成两步 安装简单的rocket环境起springboot项目测试 参考文章&#xff…

SQL Server 2012数据库允许远程连接设置

1、打开 SQL Server Management Studio 2、打开 Security 按照如下设置&#xff0c;然后点确定 3、打开SQL Server Configuration ManagerMent 4、如下图都设置为Enabled 6、sql server重启

企业和公司扩展WordPress网站的4种方法

Netflix 通过邮递观看 DVD。Apple 是一家计算机公司&#xff0c;而不是电话公司。WordPress 是一个博客平台。 这三个陈述有什么共同点&#xff1f;十年前都是对的&#xff0c;现在都不是了。如今&#xff0c;Netflix 以数字方式提供原创内容而闻名。Apple 正在推出其广受欢迎…

解决vue3中使用个别form表单校验失灵

当我点击校验时 其他都有触发校验 唯独radio没有触发&#xff0c;绑定都没有问题 看一下代码 const data reactive({form: {},rules: {serverStatus: [{ required: true, message: "服务状态不能为空", trigger: change }],tenantName: [{ required: true, messag…

Spring高手之路6——Bean生命周期的扩展点:BeanPostProcessor

文章目录 1. 探索Spring的后置处理器&#xff08;BeanPostProcessor&#xff09;1.1 BeanPostProcessor的设计理念1.2 BeanPostProcessor的文档说明 2. BeanPostProcessor的使用2.1 BeanPostProcessor的基础使用示例2.2 利用BeanPostProcessor修改Bean的初始化结果的返回值2.3 …

途乐证券|有色金属板块崛起涨超2%,云南锗业两连板

周三(7月5日)&#xff0c;A股三大股指震荡整理。截至上午收盘&#xff0c;上证指数跌幅达0.51%&#xff0c;报3228.68点&#xff1b;深证成指和创业板指跌幅分别为0.53%和0.59%&#xff1b;沪深两市合计成交额5310.1.6亿元&#xff0c;总体来看&#xff0c;两市个股跌多涨少。 …

使用Stable Diffusion生成艺术二维码

在数字艺术的世界中&#xff0c;二维码已经从单纯的信息承载工具转变为可以展示艺术表达的媒介。这是通过使用Stable Diffusion的技术实现的&#xff0c;它可以将任何二维码转化为独特的艺术作品。接下来&#xff0c;我们将一步步教你如何使用Stable Diffusion生成艺术二维码。…

【备战秋招】每日一题:2022.11.3-华为机试-去除多余空格

为了更好的阅读体检&#xff0c;可以查看我的算法学习网 在线评测链接:P1058 题目描述 塔子哥最近接到导师的一个任务&#xff0c;需要他帮忙去除文本多余空格&#xff0c;但不去除配对单引号之间的多余空格。给出关键词的起始和结束下标&#xff0c;去除多余空格后刷新关键词…

【JavaScript】ES6新特性(5)

16. Promise Promise 是异步编程的一种解决方案&#xff0c;比传统的解决方案回调函数, 更合理和更强大 ES6 将其写进了语言标准&#xff0c;统一了用法&#xff0c;原生提供了Promise对象 指定回调函数方式更灵活易懂解决异步 回调地狱 的问题 16.1 回调地狱 当一个回调函数…

并发编程 - Event Bus 设计模式

文章目录 Pre设计CodeBus接口自定义注解 Subscribe同步EventBus异步EventBusSubscriber注册表RegistryEvent广播Dispatcher 测试简单的Subscriber同步Event Bus异步Event Bus Pre 我们在日常的工作中&#xff0c;都会使用到MQ这种组件&#xff0c; 某subscriber在消息中间件上…