简单的框架
junit是由erich gamma和kent beck开发的开源测试框架,jbuilder集成了这个框架并对此做了扩展。junit之所以流行并为广大的开发人员所推崇,一是因为它实战性强,功能强大,二是因为它实在简单。一个产品或框架要能有生命力,最好都具备这样的特点。
简单地讲这个框架提供了许多断言(assert)方法,允许你设置测试的规则,如:assertequals()、assertnull()、assertnotsame()、asserttrue()等方法,一个测试用例包括了多个断言,当运行测试用例后,junit运行器会报告哪些断言没有通过,开发人员就可顺藤摸瓜搞个水落石出了。而传统的测试方法需要将期望的结果用诸如system.out.println()等语句将过程信息打印到控制台或日志中,由开发人员观察输出信息以判断是否正确,现在这种"观察"的工作由junit的那些assertxxx()方法自动完成。
junit的测试框架类结构很简单,主要由3个类组成,其类图关系如下图所示:

图 错误!文档中没有指定样式的文字。junit测试框架类结构
?junit.framework.test:测试接口。
?junit.framework.testcase:测试用例类,业务类的测试用例类只需要承继这个testcase,根据情况编写若干个public void testxxx()方法,在方法中通过assertyyy()定制若干测试规则就可以了。
?junit.framework.testsuite:测试套件类,它可以将多个测试用例类捆绑在一起运行,也可以捆绑另一个测试套件。
测试固件(fixture)
一个测试用例可以包含若干个testxxx()测试方法,测试用例测试一个或多个类api接口的正确性,当然在调用类api时,需要事先创建这个类的对象及一些关联的对象,这组对象就称为测试固件(fixture),相当于测试用例的"工作对象"。
我们在前面说过,一个测试用例类可以包含多个testxxx()方法,在运行时,每个测试方法都对应一个测试用例类的实例。当然,你可以在具体的testxxx()方法里声明并实例化业务类的实例,在测试完成后再销毁它们。但是,这么一来你就要在每个testxxx()方法中都重复这些代码,因为testcase实例被运行时,依照以下步骤运行:
1.创建测试用例的实例。
2.调用setup()方法,执行一些初始化工作。
3.运行testxxx()测试方法。
4.调用teardown()方法,执行销毁对象的工作。
如果测试用例类中有多个testxxx()方法,且它们都需要使用到相同的一组对象,我们可以在setup()中实例化这组对象,并在teardown()中销毁它们。要编写测试固件,依照以下步骤进行:
1.创建testcase类的子类。
2.在子类中声明若干个测试所用的对象。
3.覆盖setup()方法,在方法中实例化这些对象。
4.覆盖teardown()方法,释放这些对象的资源。
如下面即是一个简单的测试固件:
代码清单 错误!文档中没有指定样式的文字。测试固件
1. public class moneytest extends testcase
2. {
3. private money f12chf;//12瑞士法郎
4. private money f14chf; //14瑞士法郎
5. private money f28usd; //28美国美元
6. protected void setup() {
7. f12chf= new money(12, "chf");
8. f14chf= new money(14, "chf");
9. f28usd= new money(28, "usd");
10. }
11. protected void teardown(){}
12. }
第3~5行声明了3个money类对象(测试固件),在setup()方法中实例化这3个对象(第7~9行),由于这些对象可以被垃圾进行直接回收,所以在teardown()中不做任何操作。
测试用例(testcase)
有了测试固件,就可以开始编写测试用例的测试方法了。当然你也可不需要测试固件而直接编写测试用例方法。下面我们在测试固件的基础上添加测试用例方法testmoneybag(),代码如下所示:
代码清单 错误!文档中没有指定样式的文字。测试用例方法
1. public class moneytest extends testcase
2. {
3. private money f12chf;//12瑞士法郎
4. private money f14chf; //14瑞士法郎
5. private money f28usd; //28美国美元
6. protected void setup() {
7. f12chf= new money(12, "chf");
8. f14chf= new money(14, "chf");
9. f28usd= new money(28, "usd");
10. }
11. public void testmoneybag()
12. {
13. money bag[]= { f26chf, f28usd };
14. moneybag expected= new moneybag(bag);
15. assertequals(expected, f12chf.add(f28usd.add(f14chf)));
16. }
17. protected void teardown(){}
18. }
测试方法都必须以test为前缀,且必须是public void的,运行器以此为反射查找规则找到这些测试用例方法。在一个方法中可以包括多个assertyyy()方法,每个assertyyy()方法都是一个测试规则。像第15行的assertyyy()断言方法即为测试money的add()方法和moneybag类正确性的测试规则。
你可以在moneytest中添加多个public void testxxx()方法,运行器为每个方法生成一个测试用例实例,分别运行。
测试套件(testsuite)
如果每次只能运行一个测试用例,那么又陷入了我们前面所谈到的传统测试的窘境:手工去运行一个个测试用例,测试套件专门为解决这一问题而来。它通过testsuite对象将多个测试用例组装成到一个测试套件,则测试套件批量运行。需要特殊指出的是,可以把一个测试套件整个添加到另一个测试套件中,就象小筐装进大筐里变成一个箧一样。
测试套件类也通过承继testcase类实现,只不过它提供了一个public static test suite()静态方法,在该方法中将多个测试用例捆绑组装在一起。一个典型的测试套件代码如下所示:
代码清单 错误!文档中没有指定样式的文字。测试套件
1. public class moneytestsuite extends testcase
2. {
3. public testsuite1(string s)
4. {
5. super(s);
6. }
7. public static test suite()
8. {
9. testsuite suite = new testsuite();
10. suite.addtestsuite(moneytest.class);
11. suite.addtestsuite(moneybag.class);
12. return suite;
13. }
14. }
在第9行中声明并实例化了一个testsuite,在第10、11行分别加入一个测试用例。你可以通过suite.addtest(test t)方法添加一个套件。这样运行这个套件就可以自动运行所有测试用例的测试方法了。
测试运行器
junit提供了3个标准的测试运行器运行这些测试用例或测试套件,这3个测试运行器分别是:
?junit.textui.testrunner:文本测试运行器。
?junit.awtui.testrunner:使用awt组件界面的测试运行器。
?junit.swingui.testrunner:使用swing组件界面的测试运行器。
下面是基于awt组件的测试运行器,如下图所示:

图 错误!文档中没有指定样式的文字。awt测试运行器
?test class name:指定测试用例类和测试套件类。
?一个进度条:表示运行测试的执行进度,进度条下是正确、错误、失败的测试统计数。
?error and failures:列出了测试错误和失败的列表,点选其中的一个选项时,junit在窗口底部列出错误跟踪迹。
提示:
jbuilder提供了方便的运行测试用例和测试套件类的方法,你只须点击鼠标右键就可直接调用了。此外,jbuilder提供了一个jbtestrunner测试运行器,在功能和易用性上非junit提供的测试运行器所能媲美。jbuilder支持junit所提供的两个测试运行器:junit.textui.testrunner和junit.swingui.testrunner。
闽公网安备 35060202000074号