设计模式-组合模式(10)

组合模式定义:
        将对象以树形结构组织起来,以达成“部分-整体” 的层次结构,使得客户端对单个对象和组合对象的使用具有一致性.

组合模式理解:

        想到Composite就应该想到树形结构图。组合体内这些对象都有共同接口,当组合体一个对象的方法被调用执行时,Composite将遍历(Iterator)整个树形结构,寻找同样包含这个方法的对象并实现调用执行。可以用牵一动百来形容。

组合模式好处:

        1.使客户端调用简单,客户端可以一致的使用组合结构或其中单个对象,用户就不必关系自己处理的是单个对象还是整个组合结构,这就简化了客户端代码。
        2.更容易在组合体内加入对象部件. 客户端不必因为加入了新的对象部件而更改代码

如何使用:

        Step 1:
        首先定义一个接口或抽象类(设计模式通用方式),其他设计模式对接口内部定义限制不多,Composite却有个规定,那就是要在接口内部定义一个用于访问和管理Composite组合体对象的方法.

        Step 2:
        以抽象类或接口的方式定义一个供继承或实现的类或接口:

适用性


以下情况下适用Composite模式:

1).你想表示对象的部分-整体层次结构
2).你希望用户忽略组合对象与单个对象的不同,用户将统一地使用组合结构中的所有对象。
       

        组合模式结构图:


  组合模式的实例如下:

package com.nimiky.component;


/**
 * Created with IntelliJ IDEA.
 * User: Nimiky
 * Date: 2016/11/29 0029
 * Time: 上午 9:57
 * The Function is:测试组合模式
 */
import java.util.ArrayList;
import java.util.List;


public class ComponentDemo {
    public abstract class Component {
        String name;


        public abstract void add(Component c);


        public abstract void remove(Component c);


        public abstract void eachChild();
    }


    // 组合部件类
    public class Leaf extends Component {


        // 叶子节点不具备添加的能力,所以不实现
        @Override
        public void add(Component c) {
            // TODO Auto-generated method stub
            System.out.println("");
        }


        // 叶子节点不具备添加的能力必然也不能删除
        @Override
        public void remove(Component c) {
            // TODO Auto-generated method stub
            System.out.println("");
        }


        // 叶子节点没有子节点所以显示自己的执行结果
        @Override
        public void eachChild() {
            // TODO Auto-generated method stub
            System.out.println(name + "执行了");
        }
    }

    // 组合类
    public class Composite extends Component {

        // 用来保存节点的子节点
        List<Component> list = new ArrayList<Component>();

        // 添加节点 添加部件
        @Override
        public void add(Component c) {
            // TODO Auto-generated method stub
            list.add(c);
        }

        // 删除节点 删除部件
        @Override
        public void remove(Component c) {
            // TODO Auto-generated method stub
            list.remove(c);
        }

        // 遍历子节点
        @Override
        public void eachChild() {
            // TODO Auto-generated method stub
            System.out.println(name + "执行了");
            for (Component c : list) {
                c.eachChild();
            }
        }
    }

    public static void main(String[] args) {
        ComponentDemo demo = new ComponentDemo();
        // 构造根节点
        Composite rootComposite = demo.new Composite();
        rootComposite.name = "根节点";


        // 左节点
        Composite compositeLeft = demo.new Composite();
        compositeLeft.name = "左节点";


       // 构建右节点,添加两个叶子几点,也就是子部件
        Composite compositeRight = demo.new Composite();
        compositeRight.name = "右节点";
        Leaf leaf1 = demo.new Leaf();
        leaf1.name = "右-子节点1";
        Leaf leaf2 = demo.new Leaf();
        leaf2.name = "右-子节点2";
        compositeRight.add(leaf1);
        compositeRight.add(leaf2);


        // 左右节点加入 根节点
        rootComposite.add(compositeRight);
        rootComposite.add(compositeLeft);
        // 遍历组合部件
        rootComposite.eachChild();
    }
}


        运行结果如下:

      

      2.应用组合模式的会员卡消费

        那么我们就根据我们会员卡的消费,来模拟一下组合模式的实现吧!let's go!

        首先:

               1.我们的部件有,总店,分店,加盟店!

               2.我们的部件共有的行为是:刷会员卡

               3.部件之间的层次关系,也就是店面的层次关系是,总店下有分店、分店下可以拥有加盟店。

        有了我们这几个必要条件后,我的要求就是目前店面搞活动当我在总店刷卡后,就可以累积相当于在所有下级店面刷卡的积分总额,设计的代码如下:

 
package com.nimiky.component;

/**
 * Created with IntelliJ IDEA.
 * User: Nimiky
 * Date: 2016/11/29 0029
 * Time: 上午 10:00
 * The Function is:
 */
import java.util.ArrayList;
import java.util.List;

public class PayDemo {

    public abstract class Market {
        String name;
        public abstract void add(Market m);
        public abstract void remove(Market m);
        public abstract void PayByCard();
    }

    // 分店 下面可以有加盟店
    public class MarketBranch extends Market {
        // 加盟店列表
        List<Market> list = new ArrayList<PayDemo.Market>();
        public MarketBranch(String s) {
            this.name = s;
        }
        @Override
        public void add(Market m) {
            // TODO Auto-generated method stub
            list.add(m);
        }
        @Override
        public void remove(Market m) {
            // TODO Auto-generated method stub
            list.remove(m);
        }
        // 消费之后,该分店下的加盟店自动累加积分
        @Override
        public void PayByCard() {
            // TODO Auto-generated method stub
            System.out.println(name + "消费,积分已累加入该会员卡");
            for (Market m : list) {
                m.PayByCard();
            }
        }
    }

    // 加盟店 下面不在有分店和加盟店,最底层
    public class MarketJoin extends Market {
        public MarketJoin(String s) {
            this.name = s;
        }
        @Override
        public void add(Market m) {
            // TODO Auto-generated method stub
        }
        @Override
        public void remove(Market m) {
            // TODO Auto-generated method stub
        }
        @Override
        public void PayByCard() {
            // TODO Auto-generated method stub
            System.out.println(name + "消费,积分已累加入该会员卡");
        }
    }

    public static void main(String[] args) {
        PayDemo demo = new PayDemo();

        MarketBranch rootBranch = demo.new MarketBranch("总店");
        MarketBranch qdBranch = demo.new MarketBranch("青岛分店");
        MarketJoin snqJoin = demo.new MarketJoin("青岛分店一市南区加盟店");
        MarketJoin pdsJoin = demo.new MarketJoin("青岛分店二平度市加盟店");

        qdBranch.add(snqJoin);
        qdBranch.add(pdsJoin);
        rootBranch.add(qdBranch);
        rootBranch.PayByCard();
    }
}

      运行结果如下:

      

       这样在累积所有子店面积分的时候,就不需要去关心子店面的个数了,也不用关系是否是叶子节点还是组合节点了,也就是说不管是总店刷卡,还是加盟店刷卡,都可以正确有效的计算出活动积分。

      3.什么情况下使用组合模式

       引用大话设计模式的片段:“当发现需求中是体现部分与整体层次结构时,以及你希望用户可以忽略组合对象与单个对象的不同,统一地使用组合结构中的所有对象时,就应该考虑组合模式了。”

总结

组合模式解耦了客户程序与复杂元素内部结构,从而使客户程序可以向处理简单元素一样来处理复杂元素。

如果你想要创建层次结构,并可以在其中以相同的方式对待所有元素,那么组合模式就是最理想的选择。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值