Junit 内部解密之: Test + TestCase + TestSuite
Interface: Test 整个测试的的基础接口
Method 1: abstract int countTestCases() 这个方法主要是用来计算要运行的test case的数量的。
Method 2:abstract void run(TestResult result) 这个方法主要是用来执行一个测试用例并且在测试结果的实例中收集它的测试结果。
Class:TestCase
定义:abstract class TestCase extends Assert implements Test 继承了Assert类并且实现了Test接口。而且是个抽象类。
Class: TestSuite
定义:class TestSuite implements Test; Test接口的实现类,和TestCase一样. 但是TestSuite是一系列testcase的集合,将所有要运行的testcase add到TestSuite。
观察一:TestCase 和 TestSuite类都实现了Test 接口。
好处一:由于当你给TestSuite增加一个对象时,实际上增加的是Test,而不只是个TestCase,这样我们既可以在TestSuite里面增加另一个TestSuite,也可以加入一个TestCase。如果是TestCase,那么会运行那个单独的测试;如果是TestSuite,那么运行一组测试。
如何实现:那么是如何实现上面说的那个好处呢?看下TestSuite的一个addtest方法明白了:
Public void addTest(Test test) 新增一个test到testsuite,直接使用fTests.add(test)来新增。
这里可以新增的入参是Test接口,而不是一个TestCase对象,所以只要实现了Test接口的实现类,包括TestCase和TestSuite,都可以不断的增加下去。这样引出了好处二和设计模式。
好处二:当我们为自己的应用程序创建特殊的suite或组合出TestAll类非常容易。简单说明下TestAll类:仅仅包含了一个静态的suite方法,会注册需要定期执行的所有的Test对象(包括TestCase对象和TestSuite对象)。
Java设计模式:Composite模式
定义:把对象组合(composite)成树状结构来表示部分-整体层次关系,Composite模式可以让客户一致的对待单个对象和对象的组合。
我们来看下Junit的Test接口和TestCase和TestSuite类是如何实现Composite模式的:
先搞清楚Junit里面的部分-整体分别是什么,这里单个的TestCase可以看作是部分,把复合的TestCase(TestSuite)看作是整体,看下面的图:
我们可以看到这样的模式会带来另外一些好处:
简化了JUnit的代码 JUnit可以统一处理组合结构TestSuite和单个对象TestCase。使JUnit开发变得简单容易,因为不需要区分部分和整体的区别,不需要写一些充斥着if else的选择语句。
好处三:TestCase是个抽象实现类,而且继承了Assert类,这里为啥要继承Assert类呢?因为在一个testcase中是肯定需要用的Assert断言的,如果不继承Assert类,那么testcase方法中要写成Assert.assert(),不是很简洁,现在继承了Assert类后,我们可以在继承了TestCase类的测试类中直接assert(),使测试代码看得非常简洁和清楚。
定义了TestCase对象和TestSuite的类层次结构 基本对象TestCase可以被组合成更复杂的组合对象TestSuite,而这些组合对象又可以被组合,如我们上个例子,这样不断地递归下去。客户代码中,任何使用基本对象的地方都方便的使用组合对象,大大简化系统维护和开发。
仔细看看Test接口的方法,它存在一个是countTestCases方法,它来统计这次测试有多少个TestCase,另外一个方法run。还有一个参数TestResult,它来统计测试结果。这里为啥存在一个run方法呢? 我们想想Junit在run testcase的时候,需要把这些testcase的组成打包后成为请求发送到Junit Framework,这样我们实现TestCase类的时候,可以自由的实现run方法去打包发送请求。这样我们在写测试用例时候,只需继承TestCase,来完成run方法即可,把测试结果记录到TestResult中,这样的做法好引出另外的Java设计模式。
Java设计模式:Command模式
定义:将一个请求封装成一个对象,从而使你可用不同的请求对客户进行参数化;对请求进行排队或记录请求日志...,Command告诉我们可以为一个操作生成一个对象并给出它的一个execute(执行)方法。
run()是我们的Command模式的Excecute方法:
我们可以看到这样的模式会带来另外一些好处:
Command模式将实现请求的一方(TestCase开发)和调用一方(JUnit Fromwork)分离开。
Command模式可以将多个TestCase进行组合成一个复合命令,实际你将看到TestSuit是它的复合命令。