- 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 list = new ArrayList<>();
public void add(Component component)
{
list.add(component);
}
public void remove(Component component)
{
list.remove(component);
}
public List 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 fTests = new Vector(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 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,执行的时候会进行一次或多次递归。