- Composite,英语翻译下,复合,组合。
- 组合模式(Composite)的组成
1.Component 抽象构件接口
为组合的对象声明接口。在某些情况下实现从此接口派生出的所有类共有的默认行为。
定义一个接口可以访问及管理它的多个子部件。
2.Leaf 叶部件
在组合中表示叶节点对象,叶节点没有子节点。定义组合中接口对象的行为。
3.Composite 组合类
定义有子节点(子部件)的部件的行为。存储子节点(子部件)。
在Component接口中实现与子部件相关的操作。
4.Client 客户端
通过Component接口控制组合部件的对象。
OK,现在我们来写一个例子。
package org.linkinpark.junit.testjunit; /** * @创建作者: LinkinPark * @创建时间: 2016年2月4日 * @功能描述: 抽象构件接口。叶子类和组件类都要实现该接口 */ public interface Component { void doSomething(); }
package org.linkinpark.junit.testjunit; /** * @创建作者: LinkinPark * @创建时间: 2016年2月4日 * @功能描述: 叶子类。注意:叶子类没有子节点 */ public class Leaf implements Component { @Override public void doSomething() { System.out.println("叶子类实现。。。"); } }
package org.linkinpark.junit.testjunit; import java.util.ArrayList; import java.util.List; /** * @创建作者: LinkinPark * @创建时间: 2016年2月4日 * @功能描述: 组合类。组合类要包含子节点 */ public class Composite implements Component { // List的类型是接口类型Component,这样就既可以放Leaf,又可以放Composite private List<Component> list = new ArrayList<>(); public void add(Component component) { list.add(component); } public void remove(Component component) { list.remove(component); } public List<Component> getList() { return list; } @Override public void doSomething() { // 组合模式精髓。 for (Component component : list) { // 如果是叶子,直接执行。如果是复合的,则继续遍历其中包含的list。 component.doSomething(); } } }
package org.linkinpark.junit.testjunit; import org.junit.Test; public class CompositeTest { @Test public void testDoSomething() { Leaf leaf1 = new Leaf(); Leaf leaf2 = new Leaf(); Composite composite = new Composite(); composite.add(leaf1); composite.add(leaf2); Leaf leaf3 = new Leaf(); Leaf leaf4 = new Leaf(); Composite composite1 = new Composite(); // 组合模式来了,下面让组合类中添加2个子节点和一个组合类节点 composite1.add(leaf3); composite1.add(leaf4); composite1.add(composite); composite1.doSomething(); } }
叶子类实现。。。 叶子类实现。。。 叶子类实现。。。 叶子类实现。。。
JUnit中的测试套件Suite是一个复杂元素,但是对于用户来说,TestCase和TestSuite在使用时无需进行区分,这就是应用了组合模式。OK,现在来看看junit源码中组合模式的使用:
Test:抽象构建接口:
package org.linkinpark.commons.framework; /** * @创建作者: LinkinPark * @创建时间: 2016年1月21日 * @功能描述: 测试接口,所有的测试类都要实现这个接口 */ public interface Test { /** * @创建时间: 2016年1月21日 * @相关参数: @return * @功能描述: 测试用例执行的数量 */ public abstract int countTestCases(); /** * @创建时间: 2016年1月21日 * @相关参数: @param result * @功能描述: 开始执行一个测试用例然后+收集测试结果 */ public abstract void run(TestResult result); }
TestCase:叶子类
public abstract class TestCase extends Assert implements Test { public int countTestCases() { return 1; } public void run(TestResult result) { result.run(this); } }
TestSuite:组合类
public class TestSuite implements Test { private String fName; // 测试类的类名,注意,TestCase中的fName是方法名。 private Vector<Test> fTests = new Vector<Test>(10); // 用来装用例的,可以的是TestCase,也可以是TestSuite /** * @创建时间: 2016年1月22日 * @相关参数: @param test * @功能描述: 添加一个测试 */ public TestSuite addTest(Test test) { fTests.add(test); return this; } /** * @创建时间: 2016年1月22日 * @相关参数: @param testClass * @功能描述: 直接添加测试类到suite中 */ public TestSuite addTestSuite(Class<? extends TestCase> testClass) { addTest(new TestSuite(testClass)); return this; } public Enumeration<Test> tests() { return fTests.elements(); } /** * 统计测试用例的个数 */ public int countTestCases() { int count = 0; for (Test each : fTests) { count += each.countTestCases(); } return count; } /** * 运行测试 */ public void run(TestResult result) { for (Test each : fTests) { if (result.shouldStop()) { break; } runTest(each, result); } } public void runTest(Test test, TestResult result) { test.run(result); } }
OK,代码不多,但是利用了compiste模式。在TestSuite中,可以通过addTest()方法向TestSuite加入测试用例,也可以通过addTestSuite()方法向TestSuite加入用例集合。运行测试是通过run()方法,如果该实例是TestCase,那么执行真正的测试,如果该实例是TestSuite,那么就会通过TestSuite维护的fTests来访问每一个TestCase测试用例,然后才开始执行测试。因为这里的TestSuite也实现了Test组合类接口,所以我们可以往TestSuite中添加TestCase,也可以往TestSuite中添加TestSuite。注意:如果往TestSuite中添加的也是TestSuite,执行的时候会进行一次或多次递归。