美文网首页
组合模式

组合模式

作者: 阳光的技术小栈 | 来源:发表于2018-01-22 15:13 被阅读10次

    组合模式允许你将对象组合成树形结构来表现“整体/部分”层次结构。组合能让客户以一致的方式处理个别对象以及对象组合。

    示例—合并两种菜单并访问甜品菜单和素食菜单

    有两种餐厅菜单,分别用数组和ArrayList实现。现在需要合并两种菜单,且合并后可以让一个女招待的类去访问,不需要重复代码就可以同时访问两种菜单。 将甜品菜单添加到午餐菜单中,做到既可以访问餐厅的所有的菜单也可以单独访问甜品菜单和素食菜单。

    UML图表示

    组合模式-遍历树形菜单

    代码演示

    组件基类

    package TreeMenu;
    
    import java.util.Iterator;
    
    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 void print(){
            throw new UnsupportedOperationException();
        }
    
        public Iterator createIterator(){
            throw new UnsupportedOperationException();
        }
    }
    

    叶子菜单项

    package TreeMenu;
    
    import java.util.Iterator;
    
    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;
        }
    
        @Override
        public String toString() {
            return name + ", " + price + " -- " + description;
        }
    
        @Override
        public String getName() {
            return name;
        }
    
        @Override
        public String getDescription() {
            return description;
        }
    
        @Override
        public boolean isVegetarian() {
            return vegetarian;
        }
    
        @Override
        public double getPrice() {
            return price;
        }
    
        @Override
        public void print() {
            System.out.print(" " + getName());
            if (isVegetarian()){
                System.out.print("(v)");
            }
            System.out.println(", " + getPrice());
            System.out.println("    --" + getDescription());
        }
    
        @Override
        public Iterator createIterator() {
            return new NullIterator();
        }
    }
    

    菜单

    package TreeMenu;
    
    import java.util.ArrayList;
    import java.util.Iterator;
    
    public class Menu extends MenuComponent {
    
        ArrayList menuComponents = new ArrayList();
        String name;
        String description;
    
        public Menu(String name, String description){
            this.name = name;
            this.description = description;
        }
    
        @Override
        public void add(MenuComponent menuComponent){
            menuComponents.add(menuComponent);
        }
    
        @Override
        public void remove(MenuComponent menuComponent){
            menuComponents.remove(menuComponent);
        }
    
        @Override
        public MenuComponent getChild(int i) {
            return (MenuComponent)menuComponents.get(i);
        }
    
        @Override
        public String getName() {
            return name;
        }
    
        @Override
        public String getDescription() {
            return description;
        }
    
        @Override
        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();
            }
        }
    
        @Override
        public Iterator createIterator() {
            return new CompositeIterator(menuComponents.iterator());
        }
    
    }
    

    组合迭代器

    package TreeMenu;
    
    import java.util.Iterator;
    import java.util.Stack;
    
    public class CompositeIterator implements Iterator {
    
        Stack stack = new Stack();
    
        public CompositeIterator(Iterator iterator){
            stack.push(iterator);
        }
    
        @Override
        public boolean hasNext() {
            if (stack.empty()){
                return false;
            }
            else {
                Iterator iterator = (Iterator) stack.peek();
                if (!iterator.hasNext()) {
                    stack.pop();
                    return hasNext();
                }
                else{
                    return true;
                }
            }
        }
    
        @Override
        public Object next() {
            if (hasNext()){
                Iterator iterator = (Iterator) stack.peek();
                MenuComponent component = (MenuComponent) iterator.next();
                if (component instanceof Menu){
                    stack.push(component.createIterator());
                }
                return component;
            }
            else{
                return null;
            }
        }
    
        @Override
        public void remove() {
            throw new UnsupportedOperationException();
        }
    }
    

    空迭代器

    package TreeMenu;
    
    import java.util.Iterator;
    
    public class NullIterator implements Iterator {
        @Override
        public boolean hasNext() {
            return false;
        }
    
        @Override
        public Object next() {
            return null;
        }
    
        @Override
        public void remove() {
            throw new UnsupportedOperationException();
        }
    }
    

    服务员

    package TreeMenu;
    
    import java.util.Iterator;
    
    public class Waitress {
        MenuComponent allMenus;
    
        public Waitress(MenuComponent allMenus){
            this.allMenus = allMenus;
        }
    
        public void printMenu(){
            allMenus.print();
        }
    
        public void printVegetarianMenu(){
            Iterator iterator = allMenus.createIterator();
            System.out.println("\nVEGETARIAN MENU\n-----");
            while (iterator.hasNext()){
                MenuComponent menuComponent = (MenuComponent) iterator.next();
                try {
                    if(menuComponent.isVegetarian()){
                        menuComponent.print();
                    }
                }
                catch (UnsupportedOperationException e) {}
            }
        }
    }
    

    测试代码

    package TreeMenu;
    
    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 dessertMenu = new Menu("DESSERT MENU" , "Dessert of course!");
    
            MenuComponent allMenus = new Menu("All MENUS", "All menus combined");
    
            allMenus.add(pancakeHouseMenu);
            allMenus.add(dinerMenu);
    
            MenuItem d1 = new MenuItem("Vegetarian BLT",
                    "(Fakin')Bacon with lettuce & tomato on whole wheat", true, 2.99);
    
            MenuItem d2 = new MenuItem("Vegetarian BLT",
                    "(Fakin')Bacon with lettuce & tomato on whole wheat", true, 2.99);
            MenuItem d3 = new MenuItem("BLT","Bacon with lettuce & tomato on whole wheat", false,2.99);
            MenuItem d4 = new MenuItem("Soup of the day"
                    ,"Soup of the day, with a side of potato salad", false,3.29);
            MenuItem d5 = new MenuItem("Hotdog","A hot dog, with saurkraut, relish, onions, topped with cheese",
                    false,3.05);
    
            dinerMenu.add(d1);
            dinerMenu.add(d2);
            dinerMenu.add(d3);
            dinerMenu.add(d4);
            dinerMenu.add(d5);
    
    
            MenuItem p1 = new MenuItem("K&B's Pancake Breakfast",
                    "Pancakes with scrambled eggs, and toast",
                    true,2.99);
            MenuItem p2 = new MenuItem("Regular Pancake Breakfast",
                    "Pancakes with fired eggs, sausage",
                    false,2.99);
            MenuItem p3 = new MenuItem("Blueberry Pancakes",
                    "Pancakes made with fresh blueberries",
                    true,3.49);
            MenuItem p4 = new MenuItem("Waffles",
                    "Waffles, with your choice of blueberries or strawberries",
                    true,3.59);
    
            pancakeHouseMenu.add(p1);
            pancakeHouseMenu.add(p2);
            pancakeHouseMenu.add(p3);
            pancakeHouseMenu.add(p4);
    
    
            MenuItem ds1 = new MenuItem("Apple Pie",
                    "Apple Pie with a flaky crust",
                    true,1.59);
            MenuItem ds2 = new MenuItem("Banner Pie",
                    "Banner Pie topped with vanilla ice cream",
                    true,1.59);
            dessertMenu.add(ds1);
            dessertMenu.add(ds2);
            dinerMenu.add(dessertMenu);
    
            Waitress waitress = new Waitress(allMenus);
            waitress.printMenu();
    
            waitress.printVegetarianMenu();
        }
    }
    

    测试结果

    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 fired eggs, sausage
     Blueberry Pancakes(v), 3.49
        --Pancakes made with fresh blueberries
     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
     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
        --Soup of the day, with a side of potato salad
     Hotdog, 3.05
        --A hot dog, with saurkraut, relish, onions, topped with cheese
    
    DESSERT MENU,Dessert of course!
    -------------------------
     Apple Pie(v), 1.59
        --Apple Pie with a flaky crust
     Banner Pie(v), 1.59
        --Banner Pie topped with vanilla ice cream
    
    VEGETARIAN MENU
    -----
     K&B's Pancake Breakfast(v), 2.99
        --Pancakes with scrambled eggs, and toast
     Blueberry Pancakes(v), 3.49
        --Pancakes made with fresh blueberries
     Waffles(v), 3.59
        --Waffles, with your choice of blueberries or strawberries
     Vegetarian BLT(v), 2.99
        --(Fakin')Bacon with lettuce & tomato on whole wheat
     Vegetarian BLT(v), 2.99
        --(Fakin')Bacon with lettuce & tomato on whole wheat
     Apple Pie(v), 1.59
        --Apple Pie with a flaky crust
     Banner Pie(v), 1.59
        --Banner Pie topped with vanilla ice cream
     Apple Pie(v), 1.59
        --Apple Pie with a flaky crust
     Banner Pie(v), 1.59
        --Banner Pie topped with vanilla ice cream
    

    相关文章

      网友评论

          本文标题:组合模式

          本文链接:https://www.haomeiwen.com/subject/eumdnxtx.html