composite模式

场景: (1)父部门->子部门->子部门 (2)我们要对部门树,删除一个父部门以及其下所有的子部门

1.常规

  1. package com.example.demo.pattern.composite;
  2. import java.util.ArrayList;
  3. import java.util.List;
  4. /**
  5. * @author chenchao
  6. * @date 2021/11/9
  7. */
  8. public class WithoutCompositePatternDemo {
  9. public static void main(String[] args) {
  10. Department leafDept1 = new Department("叶子部门1");
  11. Department leafDept2 = new Department("叶子部门2");
  12. Department leafDept3 = new Department("叶子部门3");
  13. Department subDept1 = new Department("子部门1");
  14. subDept1.getChildren().add(leafDept1);
  15. subDept1.getChildren().add(leafDept2);
  16. Department subDept2 = new Department("子部门2");
  17. subDept2.getChildren().add(leafDept3);
  18. Department parentDept = new Department("父部门");
  19. parentDept.getChildren().add(subDept1);
  20. parentDept.getChildren().add(subDept2);
  21. for(Department subDept : parentDept.getChildren()) {
  22. if(subDept.getChildren().size() > 0) {
  23. for(Department leafDept : subDept.getChildren()) {
  24. leafDept.remove();
  25. }
  26. }
  27. subDept.remove();
  28. }
  29. parentDept.remove();
  30. // 问题:对层级数据的操作,很恶心,很不方便,需要手工编写大量的屎一样的代码
  31. // 平时,屎一样代码的典型特征,四五个,六七个,for if else,嵌套,嵌套的非常深,1年后,团队里没人看得懂你写的代码
  32. }
  33. public static class Department {
  34. private String name;
  35. private List<Department> children = new ArrayList<Department>();
  36. public Department(String name) {
  37. super();
  38. this.name = name;
  39. }
  40. public String getName() {
  41. return name;
  42. }
  43. public void setName(String name) {
  44. this.name = name;
  45. }
  46. public List<Department> getChildren() {
  47. return children;
  48. }
  49. public void setChildren(List<Department> children) {
  50. this.children = children;
  51. }
  52. public void remove() {
  53. System.out.println("删除部门【" + name + "】");
  54. }
  55. }
  56. }

2.组合模式

  1. package com.example.demo.pattern.composite;
  2. import java.util.ArrayList;
  3. import java.util.List;
  4. /**
  5. * @author chenchao
  6. * @date 2021/11/9
  7. */
  8. public class CompositePatternDemo {
  9. public static void main(String[] args) {
  10. Department leafDept1 = new Department("叶子部门1");
  11. Department leafDept2 = new Department("叶子部门2");
  12. Department leafDept3 = new Department("叶子部门3");
  13. Department subDept1 = new Department("子部门1");
  14. subDept1.getChildren().add(leafDept1);
  15. subDept1.getChildren().add(leafDept2);
  16. Department subDept2 = new Department("子部门2");
  17. subDept2.getChildren().add(leafDept3);
  18. Department parentDept = new Department("父部门");
  19. parentDept.getChildren().add(subDept1);
  20. parentDept.getChildren().add(subDept2);
  21. parentDept.remove();
  22. // 组合模式的第一要义,就是将树形结构的数据,用一个类,或者少数一两个类,就可以拼装成一棵树的形状
  23. // 组合模式的第二要义,就是直接对一个父级的数据执行某个操作,这个操作会直接递归调用所有下层的子数据的相关操作
  24. // 通过这个树形结构自己递归自己的方式,就将对一棵树的操作,完美的执行了
  25. // 好处,就是对树形数据的操作,不需要调用方组织复杂的屎一样的if for的代码,去执行
  26. // 外部要操作一颗树,直接对树的父级节点,调用一个操作,这颗树自己就递归着把事儿给干完了
  27. // 我跟大家这么说,武侠小说
  28. // 比较low的那种练武的人,就是只会照搬招式,一板一眼的跟着练
  29. // 然后真正的顶尖高手,到了最后,理解了武侠秘籍的思想,都是无招胜有招
  30. // 设计模式,重点,是思想,理解了思想,随便招式你怎么出,只要能将思想运用到实际业务场景中
  31. // 避免写出屎一样的代码,你就成功了
  32. // 如果你照搬设计模式去写,反而增加代码的复杂度
  33. }
  34. public static class Department {
  35. private String name;
  36. private List<Department> children = new ArrayList<Department>();
  37. public Department(String name) {
  38. super();
  39. this.name = name;
  40. }
  41. public String getName() {
  42. return name;
  43. }
  44. public void setName(String name) {
  45. this.name = name;
  46. }
  47. public List<Department> getChildren() {
  48. return children;
  49. }
  50. public void setChildren(List<Department> children) {
  51. this.children = children;
  52. }
  53. public void remove() {
  54. if(children.size() > 0) {
  55. for(Department child : children) {
  56. child.remove();
  57. }
  58. }
  59. System.out.println("删除部门【" + name + "】");
  60. }
  61. }
  62. }

3.说明

组合模式非常适用于有较多父子层级关系的那种场景,比如部门树的递归删除,或者是文件树的递归删除之类的场景。这个还是比较好模拟的,我们会在电商系统的权限那块给大家演示,比如删除一个父权限,就要级联递归删除所有的子权限,这个时候可以对组合模式做一点变种,自动从根权限开始,递归查询所有层级的子权限,然后一直到叶子节点的权限,从叶子节点开始删除