组合模式(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)成树形结构,以表示一种“部分 - 整体”的层次结构。组合让客户端(或者叫使用者)可以统一单个对象和组合对象的处理逻辑。客户端无需关心面对的是组合对象还是叶子节点对象,所以就不需要写一大堆 if-else 来保证他们对正确的对象调用了正确的方法。

不过,实现一致的处理方式意味着每一个对象都必须实现相同的接口,但由于组合中有些对象的行为不太一样,导致有些对象具备一些没有意义的方法调用,此时我们可以让这些方法不做事或者返回 null 或 false。
未命名文件 (1).jpg
举个例子,假设我们有这样一个需求:设计一个类来表示文件系统中的目录,能方便地实现下面这些功能:

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

具体代码实现如下:我们把文件和目录统一用 FileSystemNode 类来表示,并且通过 isFile 属性来区分。

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

单纯从功能实现角度来说,上面的代码没有问题,已经实现了我们想要的功能。但是,如果我们开发的是一个大型系统,从扩展性(文件或目录可能会对应不同的操作)、业务建模(文件和目录从业务上是两个概念)、代码的可读性(文件和目录区分对待更加符合人们对业务的认知)的角度来说,我们最好对文件和目录进行区分设计,定义为 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. /**
  3. * /
  4. * /wz/
  5. * /wz/a.txt
  6. * /wz/b.txt
  7. * /wz/movies/
  8. * /wz/movies/c.avi
  9. * /xzg/
  10. * /xzg/docs/
  11. * /xzg/docs/d.txt
  12. */
  13. public static void main(String[] args) {
  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. }

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

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