设计模式学习专栏十--------组合模式

设计模式学习专栏十--------组合模式

场景


回顾迭代器模式的案例

我们希望能够加上一份 餐后甜点的 "子菜单"

image

我们需要什么?

  • 我们需要某种树形结构 , 可以容纳菜单 , 子菜单和菜单项
  • 我们需要确定能够在每个菜单的各个项之间游走 . 而且至少要像现在用迭代器一样方便
  • 我们也需要能够更有弹性地在菜单项之间游走 . 比如说 , 可以只需要遍历甜品菜单, 或者可以遍历餐厅的整个菜单(包括甜点菜单在内).
image

如何实现呢?

案例结构图

image

案例类图

image

组合模式总览


定义:允许你将对象组成树形结构来表现"整体 / 部分"的层次结构.组合能让客户以一致的方式处理个别对象和对象组合(忽略整体/部分的区别)
我们可以使用这个模式来 "统一处理个别对象 和 组合对象"

  • 模式的理解

    • 类图

      image
    • 角色

      • 叶子结点Leaf
      • 组合结点 Composite
      • 叶子结点与组合结点共同接口 Component
    • 细节

      • 当我们有数个对象的集合, 它们之间有"整体/部分"的关系,并且我们想用一致的方法对待这些对象时,就可以使用组合模式

      • 组合模式提供了一个结构,可同时包容个别对象和组合对象

      • 组合结构内的任意对象为组件 , 组件可以是组合,也可以是叶子结点

      • 如果有些对象具备一些没有意义的调用(比如叶子结点的getChild), 我们可以返回null值或者false

        我们可以把没意义的调用放到父类中去默认实现 , 但这样会失去一些"安全性" (调用无效操作),

        或者把责任区分别放到不同的接口中 (比如把add , getChild放到叶子结点) , 但那样子对客户就失去了"透明性"(需要使用instanceof判断类型并强转) , 因此这需要我们折衷

案例代码部分

  • 抽象组件MenuComponent

    public abstract class MenuComponent {
       
      public void add(MenuComponent menuComponent) {
          throw new UnsupportedOperationException();
      }
      public void remove(MenuComponent menuComponent) {
          throw new UnsupportedOperationException();
      }
      public MenuComponent getChild(int i) {
          throw new UnsupportedOperationException();
      }
      
      public String getName() {
          throw new UnsupportedOperationException();
      }
      public String getDescription() {
          throw new UnsupportedOperationException();
      }
      public double getPrice() {
          throw new UnsupportedOperationException();
      }
      public boolean isVegetarian() {
          throw new UnsupportedOperationException();
      }
    
      public abstract Iterator<MenuComponent> createIterator();
     
      public void print() {
          throw new UnsupportedOperationException();
      }
    }
    
  • 菜单项MenuItem

    public class MenuItem extends MenuComponent {
     
      String name;
      String description;
      boolean vegetarian;
      double price;
        
      public MenuItem(String name, 
                      String description, 
                      boolean vegetarian, 
                      double price) 
      { 
          this.name = name;
          this.description = description;
          this.vegetarian = vegetarian;
          this.price = price;
      }
      
      public String getName() {
          return name;
      }
      
      public String getDescription() {
          return description;
      }
      
      public double getPrice() {
          return price;
      }
      
      public boolean isVegetarian() {
          return vegetarian;
      }
    
      public Iterator<MenuComponent> createIterator() {
          return new NullIterator();
      }
     
      public void print() {
          System.out.print("  " + getName());
          if (isVegetarian()) {
              System.out.print("(v)");
          }
          System.out.println(", " + getPrice());
          System.out.println("     -- " + getDescription());
      }
    
    }
    
  • 菜单Menu

    public class Menu extends MenuComponent {
      Iterator<MenuComponent> iterator = null;
      ArrayList<MenuComponent> menuComponents = new ArrayList<MenuComponent>();
      String name;
      String description;
      
      public Menu(String name, String description) {
          this.name = name;
          this.description = description;
      }
      //菜单Menu重写 add , remove ,getChild方法
      public void add(MenuComponent menuComponent) {
          menuComponents.add(menuComponent);
      }
     
      public void remove(MenuComponent menuComponent) {
          menuComponents.remove(menuComponent);
      }
     
      public MenuComponent getChild(int i) {
          return menuComponents.get(i);
      }
     
      public String getName() {
          return name;
      }
     
      public String getDescription() {
          return description;
      }
    
      
      public Iterator<MenuComponent> createIterator() {
          if (iterator == null) {
              iterator = new CompositeIterator(menuComponents.iterator());
          }
          return iterator;
      }
     
     
      public void print() {
          System.out.print("\n" + getName());
          System.out.println(", " + getDescription());
          System.out.println("---------------------");
      
          Iterator<MenuComponent> iterator = menuComponents.iterator();
          while (iterator.hasNext()) {
              MenuComponent menuComponent = iterator.next();
              menuComponent.print();
          }
      }
    }
    
  • 客户端

    public class MenuTestDrive {
      public static void main(String args[]) {
    
          MenuComponent pancakeHouseMenu = 
              new Menu("PANCAKE HOUSE MENU", "Breakfast");
          MenuComponent dinerMenu = 
              new Menu("DINER MENU", "Lunch");
      
          MenuComponent allMenus = new Menu("ALL MENUS", "All menus combined");
      
          allMenus.add(pancakeHouseMenu);
          allMenus.add(dinerMenu);
      
          pancakeHouseMenu.add(new MenuItem(
              "K&B's Pancake Breakfast", 
              "Pancakes with scrambled eggs, and toast", 
              true,
              2.99));
          pancakeHouseMenu.add(new MenuItem(
              "Regular Pancake Breakfast", 
              "Pancakes with fried eggs, sausage", 
              false,
              2.99));
          pancakeHouseMenu.add(new MenuItem(
              "Blueberry Pancakes",
              "Pancakes made with fresh blueberries, and blueberry syrup",
              true,
              3.49));
          pancakeHouseMenu.add(new MenuItem(
              "Waffles",
              "Waffles, with your choice of blueberries or strawberries",
              true,
              3.59));
    
          dinerMenu.add(new MenuItem(
              "Vegetarian BLT",
              "(Fakin') Bacon with lettuce & tomato on whole wheat", 
              true, 
              2.99));
          dinerMenu.add(new MenuItem(
              "BLT",
              "Bacon with lettuce & tomato on whole wheat", 
              false, 
              2.99));
          dinerMenu.add(new MenuItem(
              "Soup of the day",
              "A bowl of the soup of the day, with a side of potato salad", 
              false, 
              3.29));
          dinerMenu.add(new MenuItem(
              "Hotdog",
              "A hot dog, with saurkraut, relish, onions, topped with cheese",
              false, 
              3.05));
          dinerMenu.add(new MenuItem(
              "Steamed Veggies and Brown Rice",
              "A medly of steamed vegetables over brown rice", 
              true, 
              3.99));
     
          dinerMenu.add(new MenuItem(
              "Pasta",
              "Spaghetti with Marinara Sauce, and a slice of sourdough bread",
              true, 
              3.89));
    
          Waitress waitress = new Waitress(allMenus);
          waitress.printMenu();
      }
    }
    
  • 输出结果

    ALL MENUS, All menus combined
    ---------------------
    
    PANCAKE HOUSE MENU, Breakfast
    ---------------------
      K&B's Pancake Breakfast(v), 2.99
         -- Pancakes with scrambled eggs, and toast
      Regular Pancake Breakfast, 2.99
         -- Pancakes with fried eggs, sausage
      Blueberry Pancakes(v), 3.49
         -- Pancakes made with fresh blueberries, and blueberry syrup
      Waffles(v), 3.59
         -- Waffles, with your choice of blueberries or strawberries
    
    DINER MENU, Lunch
    ---------------------
      Vegetarian BLT(v), 2.99
         -- (Fakin') Bacon with lettuce & tomato on whole wheat
      BLT, 2.99
         -- Bacon with lettuce & tomato on whole wheat
      Soup of the day, 3.29
         -- A bowl of the soup of the day, with a side of potato salad
      Hotdog, 3.05
         -- A hot dog, with saurkraut, relish, onions, topped with cheese
      Steamed Veggies and Brown Rice(v), 3.99
         -- A medly of steamed vegetables over brown rice
      Pasta(v), 3.89
         -- Spaghetti with Marinara Sauce, and a slice of sourdough bread
    

参考

​ 书籍: HeadFirst设计模式

​ 代码参考地址: 我就是那个地址

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,717评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,501评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,311评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,417评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,500评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,538评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,557评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,310评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,759评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,065评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,233评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,909评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,548评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,172评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,420评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,103评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,098评论 2 352

推荐阅读更多精彩内容