當 JUnit 不必區分其運行的是一個或多個測試用例時,能夠輕松地解決這個問題的模式就
是 Composite(組合)模式。摘引其意圖,"將對象組合成樹形結構以表示'部分-整體'的層次
結構。Composite 使得用戶對單個對象和組合對象的使用具有一致性。"在這里'部分-整體'
的層次結構是解決問題的關鍵,可以把單個的 TestCase 看作部分,而把復合的 TestCase 看
作整體(稱為 TestSuit) 。這樣使用該模式便可以恰到好處得解決了這個難題。
Composite 模式引入以下的參與者:
? Component:這是一個抽象角色,它給參加組合的對象規定一個接口。這個角色,
給出共有的接口和默認行為。其實就我們的 Test 接口,它定義出 run 方法。
? Composite:實現共有接口并維護一個測試用例的集合。就是復合測試用例 TestSuit
? Leaf:代表參加組合的對象,它沒有下級子對象,僅定義出參加組合的原始對象的
行為,其實就是單一的測試用例 TestCase,它僅實現 Test 接口的方法。
其實 componsite 模式根據所實現的接口類型區分為兩種形式,分別稱為安全式和透明式。
JUnit 中使用了安全式的結構,這樣在 TestCase 中沒有管理子對象的方法。
composite 模式告訴我們要引入一個 Component 抽象類,為 Leaf 對象和 composite 對象
定義公共的接口。這個類的基本意圖就是定義一個接口。在 Java 中使用 Composite模式時,
優先考慮使用接口,而非抽象類,因此引入一個 Test 接口。當然我們的 leaf 就是 TestCase
了。其源代碼如下:
//composite模式中的Component角色
public interface Test {
public abstract void run(TestResult result);
}
//composite模式中的Leaf角色
public abstract class TestCase extends Assert implements Test {
public void run(TestResult result) {
result.run(this);
}
}
下面,列出 Composite 源碼。將其取名為 TestSuit 類。TestSuit 有一個屬性 fTests (Vector
類型)中保存了其子測試用例,提供 addTest 方法來實現增加子對象 TestCase ,并且還提供
estCount 和 tests 等方法來操作子對象。最后通過 run()方法實現對其子對象進行委托
(delegate) ,最后還提供 addTestSuite 方法實現遞歸,構造成樹形。
public class TestSuite implements Test {
private Vector fTests= new Vector(10);
public void addTest(Test test) {
fTests.addElement(test);
}
public Enumeration tests() {
return fTests.elements();
}
public void run(TestResult result) {
for (Enumeration e= tests(); e.hasMoreElements(); ) {
Test test= (Test)e.nextElement();
runTest(test, result);
}
}
public void addTestSuite(Class testClass) {
addTest(new TestSuite(testClass));
}
}
注意所有上面的代碼是對 Test 接口進行實現的。 由于 TestCase 和 TestSuit兩者都符合 Test
接口,我們可以通過 addTestSuite 遞歸地將 TestSuite 再組合成 TestSuite,這樣將構成樹形結
構。所有開發者都能夠創建他們自己的 TestSuit。測試人員可創建一個組合了這些測試用例
的 TestSuit 來運行它們所有的 TestCase。
public static Test suite() {
TestSuite suite1 = new TestSuite("我的測試TestSuit1");
TestSuite suite2 = new TestSuite("我的測試TestSuit2");
suite1.addTestSuite(untitled6.Testmath.class);
suite2.addTestSuite(untitled6.Testmulti.class);
suite1.addTest(suite2);
return suite1;
}