结构型设计模式就快要讲完了,还剩下两个不那么常用的:组合模式和享元模式。
今天,我们来讲一下组合模式(Composite Design Pattern)。
组合模式跟我们之前讲的面向对象设计中的“组合关系(通过组合来组装两个类)”,完全是两码事。
这里讲的“组合模式”,主要是用来处理树形结构数据。这里的“数据”,你可以简单理解为一组对象集合,待会我们会详细讲解。
正因为其应用场景的特殊性,数据必须能表示成树形结构,这也导致了这种模式在实际的项目开发中并不那么常用。但是,一旦数据满足树形结构,应用这种模式就能发挥很大的作用,能让代码变得非常简洁。
话不多说,让我们正式开始今天的学习吧!

组合模式的原理与实现

在 GoF 的《设计模式》一书中,组合模式是这样定义的:Compose objects into tree structure to represent part-whole hierarchies.Composite lets client treat individual objects and compositions of objects uniformly.
翻译成中文就是:将一组对象组织(Compose)成树形结构,以表示一种“部分 - 整体”的层次结构。组合让客户端(在很多设计模式书籍中,“客户端”代指代码的使用者。)可以统一单个对象和组合对象的处理逻辑。


接下来,对于组合模式,我举个例子来给你解释一下。
假设我们有这样一个需求:设计一个类来表示文件系统中的目录,能方便地实现下面这些功能:

  • 动态地添加、删除某个目录下的子目录或文件;
  • 统计指定目录下的文件个数;
  • 统计指定目录下的文件总大小。

我这里给出了这个类的骨架代码,如下所示。
其中的核心逻辑并未实现,你可以试着自己去补充完整,再来看我的讲解。
在下面的代码实现中,把文件和目录统一用 FileSystemNode 类来表示,并且通过 isFile 属性来区分。

  1. // 表示文件和目录
  2. public class FileSystemNode {
  3. private String path;
  4. // 区分文件和目录
  5. private boolean isFile;
  6. private List<FileSystemNode> subNodes = new ArrayList<>();
  7. public FileSystemNode(String path, boolean isFile) {
  8. this.path = path;
  9. this.isFile = isFile;
  10. }
  11. public int countNumOfFiles() {
  12. // TODO:...
  13. }
  14. public long countSizeOfFiles() {
  15. // TODO:...
  16. }
  17. public String getPath() {
  18. return path;
  19. }
  20. public void addSubNode(FileSystemNode fileOrDir) {
  21. subNodes.add(fileOrDir);
  22. }
  23. public void removeSubNode(FileSystemNode fileOrDir) {
  24. int size = subNodes.size();
  25. int i = 0;
  26. for (; i < size; ++i) {
  27. if (subNodes.get(i).getPath().equalsIgnoreCase(fileOrDir.getPath())) {
  28. break;
  29. }
  30. }
  31. if (i < size) {
  32. subNodes.remove(i);
  33. }
  34. }
  35. }

实际上,如果你看过我的《数据结构与算法之美》专栏,想要补全其中的 countNumOfFiles() 和 countSizeOfFiles() 这两个函数,并不是件难事,实际上这就是树上的递归遍历算法。

  • 对于文件,我们直接返回文件的个数(返回 1)或大小。
  • 对于目录,我们遍历目录中每个子目录或者文件,递归计算它们的个数或大小,然后求和,就是这个目录下的文件个数和文件大小。

我把两个函数的代码实现贴在下面了,你可以对照着看一下。

  1. public int countNumOfFiles() {
  2. if (isFile) {
  3. return 1;
  4. }
  5. int numOfFiles = 0;
  6. for (FileSystemNode fileOrDir : subNodes) {
  7. numOfFiles += fileOrDir.countNumOfFiles();
  8. }
  9. return numOfFiles;
  10. }
  11. public long countSizeOfFiles() {
  12. if (isFile) {
  13. File file = new File(path);
  14. if (!file.exists()) return 0;
  15. return file.length();
  16. }
  17. long sizeofFiles = 0;
  18. for (FileSystemNode fileOrDir : subNodes) {
  19. sizeofFiles += fileOrDir.countSizeOfFiles();
  20. }
  21. return sizeofFiles;
  22. }

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

  1. public abstract class FileSystemNode {
  2. protected String path;
  3. public FileSystemNode(String path) {
  4. this.path = path;
  5. }
  6. public abstract int countNumOfFiles();
  7. public abstract long countSizeOfFiles();
  8. public String getPath() {
  9. return path;
  10. }
  11. }
  12. public class File extends FileSystemNode {
  13. public File(String path) {
  14. super(path);
  15. }
  16. @Override
  17. public int countNumOfFiles() {
  18. return 1;
  19. }
  20. @Override
  21. public long countSizeOfFiles() {
  22. java.io.File file = new java.io.File(path);
  23. if (!file.exists()) return 0;
  24. return file.length();
  25. }
  26. }
  27. public class Directory extends FileSystemNode {
  28. private List<FileSystemNode> subNodes = new ArrayList<>();
  29. public Directory(String path) {
  30. super(path);
  31. }
  32. @Override
  33. public int countNumOfFiles() {
  34. int numOfFiles = 0;
  35. for (FileSystemNode fileOrDir : subNodes) {
  36. numOfFiles += fileOrDir.countNumOfFiles();
  37. }
  38. return numOfFiles;
  39. }
  40. @Override
  41. public long countSizeOfFiles() {
  42. long sizeofFiles = 0;
  43. for (FileSystemNode fileOrDir : subNodes) {
  44. sizeofFiles += fileOrDir.countSizeOfFiles();
  45. }
  46. return sizeofFiles;
  47. }
  48. public void addSubNode(FileSystemNode fileOrDir) {
  49. subNodes.add(fileOrDir);
  50. }
  51. public void removeSubNode(FileSystemNode fileOrDir) {
  52. int size = subNodes.size();
  53. int i = 0;
  54. for (; i < size; ++i) {
  55. if (subNodes.get(i).getPath().equalsIgnoreCase(fileOrDir.getPath())) {
  56. break;
  57. }
  58. }
  59. if (i < size) {
  60. subNodes.remove(i);
  61. }
  62. }
  63. }

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

  1. public class Demo {
  2. public static void main(String[] args) {
  3. /**
  4. * /
  5. * /wz/
  6. * /wz/a.txt
  7. * /wz/b.txt
  8. * /wz/movies/
  9. * /wz/movies/c.avi
  10. * /xzg/
  11. * /xzg/docs/
  12. * /xzg/docs/d.txt
  13. */
  14. Directory fileSystemTree = new Directory("/");
  15. Directory node_wz = new Directory("/wz/");
  16. Directory node_xzg = new Directory("/xzg/");
  17. fileSystemTree.addSubNode(node_wz);
  18. fileSystemTree.addSubNode(node_xzg);
  19. File node_wz_a = new File("/wz/a.txt");
  20. File node_wz_b = new File("/wz/b.txt");
  21. Directory node_wz_movies = new Directory("/wz/movies/");
  22. node_wz.addSubNode(node_wz_a);
  23. node_wz.addSubNode(node_wz_b);
  24. node_wz.addSubNode(node_wz_movies);
  25. File node_wz_movies_c = new File("/wz/movies/c.avi");
  26. node_wz_movies.addSubNode(node_wz_movies_c);
  27. Directory node_xzg_docs = new Directory("/xzg/docs/");
  28. node_xzg.addSubNode(node_xzg_docs);
  29. File node_xzg_docs_d = new File("/xzg/docs/d.txt");
  30. node_xzg_docs.addSubNode(node_xzg_docs_d);
  31. System.out.println("/ files num:" + fileSystemTree.countNumOfFiles());
  32. System.out.println("/wz/ files num:" + node_wz.countNumOfFiles());
  33. }
  34. }

我们对照着这个例子,再重新看一下组合模式的定义:“将一组对象(文件和目录)组织成树形结构,以表示一种‘部分 - 整体’的层次结构(目录与子目录的嵌套结构)。组合模式让客户端可以统一单个对象(文件)和组合对象(目录)的处理逻辑(递归遍历)。”
实际上,刚才讲的这种组合模式的设计思路,与其说是一种设计模式,倒不如说是对业务场景的一种数据结构和算法的抽象。其中,数据可以表示成树这种数据结构,业务需求可以通过在树上的递归遍历算法来实现。

组合模式的应用场景举例

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


假设我们在开发一个 OA 系统(办公自动化系统)。
公司的组织结构包含部门和员工两种数据类型。
其中,部门又可以包含子部门和员工。
在数据库中的表结构如下所示:
image.png
我们希望在内存中构建整个公司的人员架构图(部门、子部门、员工的隶属关系),并且提供接口计算出部门的薪资成本(隶属于这个部门的所有员工的薪资和)。
部门包含子部门和员工,这是一种嵌套结构,可以表示成树这种数据结构。计算每个部门的薪资开支这样一个需求,也可以通过在树上的遍历算法来实现。所以,从这个角度来看,这个应用场景可以使用组合模式来设计和实现。
这个例子的代码结构跟上一个例子的很相似,代码实现我直接贴在了下面,你可以对比着看一下。
其中,HumanResource 是部门类(Department)和员工类(Employee)抽象出来的父类,为的是能统一薪资的处理逻辑。
Demo 中的代码负责从数据库中读取数据并在内存中构建组织架构图。

  1. public abstract class HumanResource {
  2. protected long id;
  3. protected double salary;
  4. public HumanResource(long id) {
  5. this.id = id;
  6. }
  7. public long getId() {
  8. return id;
  9. }
  10. public abstract double calculateSalary();
  11. }
  12. public class Employee extends HumanResource {
  13. public Employee(long id, double salary) {
  14. super(id);
  15. this.salary = salary;
  16. }
  17. @Override
  18. public double calculateSalary() {
  19. return salary;
  20. }
  21. }
  22. public class Department extends HumanResource {
  23. private List<HumanResource> subNodes = new ArrayList<>();
  24. public Department(long id) {
  25. super(id);
  26. }
  27. @Override
  28. public double calculateSalary() {
  29. double totalSalary = 0;
  30. for (HumanResource hr : subNodes) {
  31. totalSalary += hr.calculateSalary();
  32. }
  33. this.salary = totalSalary;
  34. return totalSalary;
  35. }
  36. public void addSubNode(HumanResource hr) {
  37. subNodes.add(hr);
  38. }
  39. }
  40. // 构建组织架构的代码
  41. public class Demo {
  42. private static final long ORGANIZATION_ROOT_ID = 1001;
  43. private DepartmentRepo departmentRepo; // 依赖注入
  44. private EmployeeRepo employeeRepo; // 依赖注入
  45. public void buildOrganization() {
  46. Department rootDepartment = new Department(ORGANIZATION_ROOT_ID);
  47. buildOrganization(rootDepartment);
  48. }
  49. private void buildOrganization(Department department) {
  50. List<Long> subDepartmentIds = departmentRepo.getSubDepartmentIds(department.getId());
  51. for (Long subDepartmentId : subDepartmentIds) {
  52. Department subDepartment = new Department(subDepartmentId);
  53. department.addSubNode(subDepartment);
  54. buildOrganization(subDepartment);
  55. }
  56. List<Long> employeeIds = employeeRepo.getDepartmentEmployeeIds(department.getId());
  57. for (Long employeeId : employeeIds) {
  58. double salary = employeeRepo.getEmployeeSalary(employeeId);
  59. department.addSubNode(new Employee(employeeId, salary));
  60. }
  61. }
  62. }

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

重点回顾

好了,今天的内容到此就讲完了。我们一块来总结回顾一下,你需要重点掌握的内容。
组合模式的设计思路,与其说是一种设计模式,倒不如说是对业务场景的一种数据结构和算法的抽象。其中,数据可以表示成树这种数据结构,业务需求可以通过在树上的递归遍历算法来实现。
组合模式,将一组对象组织成树形结构,将单个对象和组合对象都看做树中的节点,以统一处理逻辑,并且它利用树形结构的特点,递归地处理每个子树,依次简化代码实现。
使用组合模式的前提在于,你的业务场景必须能够表示成树形结构。所以,组合模式的应用场景也比较局限,它并不是一种很常用的设计模式。

课堂讨论

在文件系统那个例子中,countNumOfFiles() 和 countSizeOfFiles() 这两个函数实现的效率并不高,因为每次调用它们的时候,都要重新遍历一遍子树。
有没有什么办法可以提高这两个函数的执行效率呢(注意:文件系统还会涉及频繁的删除、添加文件操作,也就是对应 Directory 类中的 addSubNode() 和 removeSubNode() 函数)?