组合模式 Composite

组合模式Composite

  组合模式有时候又叫做部分-整体模式,它使我们在树形结构的问题中,模糊了简单元素和复杂元素的概念,客户程序可以像处理简单元素一样来处理复杂元素,从而使得客户与复杂元素的内部结构解耦。

组合模式的意图

  将对象组合成树形结构以表示“部分-整体”的层次结构。

  Composite模式使得用户对单个对象和组合对象的使用具有一致性。

 

组合模式(Composite)的组成

1.Component 抽象构件接口

  为组合的对象声明接口。

  在某些情况下实现从此接口派生出的所有类共有的默认行为。

  定义一个接口可以访问及管理它的多个子部件。

2.Leaf 叶部件

  在组合中表示叶节点对象,叶节点没有子节点。

  定义组合中接口对象的行为。

3.Composite 组合类

  定义有子节点(子部件)的部件的行为。

  存储子节点(子部件)。

  在Component接口中实现与子部件相关的操作。

4.Client 客户端

  通过Component接口控制组合部件的对象。

在JUnit中的应用

  JUnit中的测试套件Suite是一个复杂元素,但是对于用户来说,TestCase和TestSuite在使用时无需进行区分,这就是应用了组合模式。

组合模式的实现

  组合模式有两种实现方式:

  1.将管理子元素的方法定义在Composite中。

  2.将管理子元素的方法定义在Component接口中。这样Leaf类就要对这些方法空实现。

组合模式的第一种实现

复制代码
package com.meng.designpattern.composite;

//接口:复杂对象和简单对象都实现这个接口,所以外部调用时可相同看待
public interface Component
{
    public void doSomething();

}
复制代码

  叶子节点:

复制代码
package com.meng.designpattern.composite;

public class Leaf implements Component
{

    @Override
    public void doSomething()
    {
        System.out.println("执行方法 -- in Leaf: " + this.toString());

    }

}
复制代码

  复杂节点:

复制代码
package com.meng.designpattern.composite;

import java.util.List;
import java.util.ArrayList;

public class Composite implements Component
{
    // List的类型是接口类型Component,这样就既可以放Leaf,又可以放Composite
    private List<Component> list = new ArrayList<Component>();

    public void add(Component component)
    {
        list.add(component);
    }

    public void remove(Component component)
    {
        list.remove(component);
    }

    public List<Component> getAll()
    {
        return this.list;
    }

    @Override
    public void doSomething()
    {
        for (Component component : list)
        {
            // 如果是叶子,直接执行
            // 如果是复合的,则继续遍历其中包含的list
            component.doSomething();
        }

    }

}
复制代码

  运用:

复制代码
package com.meng.designpattern.composite;

public class Client
{
    public static void main(String[] args)
    {
        Component leaf1 = new Leaf();
        Component leaf2 = new Leaf();

        Composite composite1 = new Composite();
        composite1.add(leaf1);
        composite1.add(leaf2);

        Component leaf3 = new Leaf();
        Component leaf4 = new Leaf();

        Composite composite2 = new Composite();
        composite2.add(composite1);
        composite2.add(leaf3);
        composite2.add(leaf4);

        // Composite和Leaf执行起来无差别:
        composite2.doSomething();
        leaf4.doSomething();
    }

}
复制代码

组合模式的第二种实现

   将接口如下定义:

复制代码
package com.meng.designpattern.composite2;

import java.util.List;

//这样外部接口可以全是Component
public interface Component
{
    public void doSomething();

    public void add(Component component);

    public void remove(Component component);

    public List<Component> getAll();

}
复制代码

  复合节点的定义如前:

复制代码
package com.meng.designpattern.composite2;

import java.util.ArrayList;
import java.util.List;

public class Composite implements Component
{
    private List<Component> list = new ArrayList<Component>();

    @Override
    public void doSomething()
    {
        for (Component component : list)
        {
            component.doSomething();
        }

    }

    @Override
    public void add(Component component)
    {
        list.add(component);

    }

    @Override
    public void remove(Component component)
    {
        list.remove(component);

    }

    @Override
    public List<Component> getAll()
    {
        return this.list;
    }

}
复制代码

  但是叶子节点比前面复杂,需要实现接口的全部方法,提供一些空实现:

复制代码
package com.meng.designpattern.composite2;

import java.util.List;

public class Leaf implements Component
{

    @Override
    public void doSomething()
    {
        System.out.println("执行方法 -- in Leaf: " + this.toString());

    }

    // 都是空实现
    @Override
    public void add(Component component)
    {

    }

    @Override
    public void remove(Component component)
    {

    }

    @Override
    public List<Component> getAll()
    {
        return null;
    }

}
复制代码

  客户端调用代码类似,不再重复。这里要注意用第二种实现方法实现时,客户端调用时所有的引用类型都可以用接口类型,因为接口中包含了所有要调用的方法。

参考资料

  圣思园张龙老师视频教程。

原文地址:https://www.cnblogs.com/huangcongcong/p/4749616.html