- 浏览: 131647 次
- 性别:
- 来自: 广州
文章分类
最新评论
-
jingjieyiman:
引用
[url][/url][flash=200,200][ ...
Java 学习入门到高深 -
Technoboy:
...
AOP 的利器:ASM 3.0 介绍 -
kingkan:
非常支持,我有意愿加入。。。如果可以,请给我短消息,je我每天 ...
【征集】做一个管理人际关系的应用 -
夜之son:
csuzm0613 写道能否直接做成Android应用程序呢? ...
【征集】做一个管理人际关系的应用 -
gdwrx_winson:
谢谢异常哥的回复,目前的想法是做成一个web应用抛出异常的爱 ...
【征集】做一个管理人际关系的应用
转自http://blog.csdn.net/ai92/archive/2005/03/12/318318.aspx
JUnit源码分析(一) 收藏
一、引子
JUnit源码是我仔细阅读过的第一个开源项目源码。阅读高手写的代码能学到一些好的编程风格和实现思路,这是提高自己编程水平行之有效的方法,因此早就想看看这些赫赫有名的框架是怎么回事了。今天就拿最简单的JUnit下手,也算开始自己的源码分析之路。
JUnit作为最著名的单元测试框架,由两位业界有名人士协力完成,已经经历了多次版本升级(了解JUnit基础、JUnit实践)。JUnit总体来说短小而精悍,有不少值得我们借鉴的经验在里面;但是也有一些不足存在,当然这对于任何程序来说都是难免的。
下面我们将从整体(宏观)和细节(微观)两方面来分析JUnit源码,以下分析基于3.8.1版。
二、宏观——架构与模式
打开源码文件,你会发现JUnit源码被分配到6个包中:junit.awtui、junit.swingui、junit.textui、junit.extensions、junit.framework、junit.runner。其中前三个包中包含了JUnit运行时的入口程序以及运行结果显示界面,它们对于JUnit使用者来说基本是透明的。junit.runner包中包含了支持单元测试运行的一些基础类以及自己的类加载器,它对于JUnit使用者来说是完全透明的。
剩下的两个包是和使用JUnit进行单元测试紧密联系在一起的。其中junit.framework包含有编写一般JUnit单元测试类必须是用到的JUnit类;而junit.extensions则是对framework包在功能上的一些必要扩展以及为更多的功能扩展留下的接口。
JUnit提倡单元测试的简单化和自动化。这就要求JUnit的使用要简单化,而且要很容易的实现自动化测试。整个JUnit的设计大概也是遵循这个前提吧。整个框架的骨干仅有三个类组成(下图所示)。
如果你掌握了TestCase、TestSuite、BaseTestRunner的工作方式,那么你就可以随心所欲的编写测试代码了。
下面我们来看看junit.framework中类之间的关系,下图是我根据源代码分析出来的,大部分关系都表示了出来。
先来看看各个类的职责。Assert类提供了JUnit使用的一整套的断言,这套断言都被TestCase继承下来,Assert也就变成了透明的。Test接口是为了统一TestCase和TestSuite的类型;而TestCase里面提供了运行单元测试类的方法;在TestSuite中则提供了加载单元测试类,检验测试类格式等等的方法。TestResult故名思意就是提供存放测试结果的地方,但是在JUnit中它还带有一点控制器的功能。TestListener接口抽象了所有测试监听者的行为,他包括两个添加错误和失败的方法,开始测试和结束测试的方法。在JUnit框架中有两个类实现了这个接口,一个负责结果打印的ResultPrinter类,一个是所有TestRunner的基础类BaseTestRunner类(这两个类都不在framework包中)。
在这里指出其中我认为有些不妥的地方。图上TestCase和TestResult之间是双向的依赖关系,而在UML类图的关系中指出:依赖关系总是单向的。就让我们来看看这这个可疑的地方。
TestCase中的代码:
/**
* Runs the test case and collects the results in TestResult.
*/
public void run(TestResult result) {
//调用了result中的run方法,
//TestResult按照名称来看应该是一个记录测试结果的类,怎么还能run?
result.run(this);
}
相应得TestResult中的代码:
/**
* Runs a TestCase.
*/
protected void run(final TestCase test) {
//开始测试
startTest(test);
//这个匿名内类的使用一会再讲
Protectable p= new Protectable() {
public void protect() throws Throwable {
//天那,这里又调用了TestCase里面的runBare方法
test.runBare();
}
};
runProtected(test, p); //这个方法就是要执行上面制定的匿名内类
endTest(test);
}
TestResult中runProtected方法:
public void runProtected(final Test test, Protectable p) {
try {
p.protect();
}
catch (AssertionFailedError e) {
addFailure(test, e); //给TestResult添加失败记录
}
catch (ThreadDeath e) { // don't catch ThreadDeath by accident
throw e;
}
catch (Throwable e) {
addError(test, e); //给TestResult添加出错记录
}
}
为什么JUnit里面会出现这样奇怪的依赖关系,还有违反单一职责原则的TestResult?当我看到junit.extentions包中的TestSetup时,也许我猜到了作者的用意。我们来看下TestSetup中有关的代码:
public void run(final TestResult result) {
//又看到了上面类似的匿名内部类
Protectable p= new Protectable() {
public void protect() throws Exception {
//不过这个内部类里面的实现有所不同
setUp();
basicRun(result);
tearDown();
}
};
//调用了TestResult中的runProtected方法来执行上面的实现
result.runProtected(this, p);
}
这个类的产生是为了弥补TestCase类的一个小小的缺陷(具体请见下部分)。注意到在这个类里面也有和TestResult类似的匿名内部类。这种匿名内部类全是Protected接口的无名实现,这里的目的我认为有两点:
1) 由于内部类可以在接下来的情景中完全不可见,而且不被任何人使用,因此也就隐藏了接口的实现细节。
2) 为了提高可重用性,而使用内部类比较快捷。这样不管你protect方法里面具体执行什么,对它错误、失败、异常捕捉的代码(TestResult中的runProtected方法)就可以重用了。
这也正是为什么会出现上面那样奇怪的依赖关系:为了复用,就要让runProtected方法放在一个TestCase和TestSetup都能调用的地方。
不过我认为为了复用而破坏了系统良好的结构和可读性,是需要仔细斟酌的。JUnit这样的设计估计是为了以后框架多次扩展后的重用考虑的。
说完了让我费解的问题。谈谈我觉得JUnit框架中最让我感叹的地方,那就是小小的框架里面使用了很多设计模式在里面。而这些模式的使用也正是为了体现出整个框架结构的简洁、可扩展。我将粗略的分析如下(模式应用的详细内容请关注我关于设计模式的文章)。先看看在junit.framework里面使用的设计模式。
命令模式:作为辅助单元测试的框架,开发人员在使用它的时候,应该仅仅关心测试用例的编写,JUnit只是一个测试用例的执行器和结果查看器,不应该关心太多关于这个框架的细节。而对于JUnit来说,它并不需要知道请求TestCase的操作信息,仅把它当作一种命令来执行,然后把执行测试结果发给开发人员。命令模式正是为了达到这种送耦合的目的。
组合模式:当系统的测试用例慢慢变得多起来,挨个运行测试用例就成了一个棘手的问题。作为一个方便使用的单元测试框架,这一点是必须解决的。因此JUnit里面提供了TestSuite的功能,它允许将多个测试用例放到一个TestSuite里面来一次执行;而且要进一步的支持TestSuite里面套TestSuite的功能。使用组合模式能够很好的解决这个问题。
模板模式:JUnit在TestCase这个抽象类中将整个测试的流程设置好了,比如先执行Setup方法初始化测试前提,在运行测试方法,然后再TearDown来取消测试设置。而这些步骤的具体实现都延迟到子类中去,也就是你实现的测试类中。
模板模式:JUnit在TestCase这个抽象类中将整个测试的流程设置好了,比如先执行Setup方法初始化测试前提,在运行测试方法,然后再TearDown来取消测试设置。而这些步骤的具体实现都延迟到子类中去,也就是你实现的测试类中。
发表评论
-
【征集】做一个管理人际关系的应用
2011-03-07 21:10 1167背景:目前人际交往越来越密切,但是同时处理好庞大的人际关系又比 ... -
输入输出流的复制
2010-12-30 09:26 666public static void copy(Inpu ... -
RAD快速开发
2010-12-14 12:28 1328RAD=rapid application develop(快 ... -
TreePath的用法
2010-12-14 10:53 1387TreeSelection sel = (TreeSelect ... -
获取数据库中存储过程名
2010-12-14 10:49 857获取数据库中存储过程名 This example retri ... -
数字输入校验
2010-12-14 10:48 7701. txt_TimeLimit = new T ... -
GMF手动增加一个EditPart需要修改代码
2010-12-14 10:19 978edit: 创建相应的command、helper、edi ... -
Collections
2010-12-14 10:16 1101Collections.unmodifiableSet() ... -
Collections
2010-12-14 10:16 1000Collections.unmodifiableSet() ... -
JMX连接JConsole字符串
2010-12-14 10:13 973service:jmx:iiop:///jndi/corban ... -
Problem视图增加Marker
2010-12-14 10:00 1322首先,需要得到Problems视图窗口: IWorkbenc ... -
SQL数据库类型
2010-12-14 09:59 839SQL Server 2000 Data Types SQL ... -
数据库相关知识
2010-12-14 09:22 730String strurl = "jdbc:o ... -
FileOutputStream
2010-12-10 14:00 1293FileOutputStream可以创建文件,但一般使用new ... -
关于端口被占用
2010-11-12 10:49 827今天整了一下JBoss的ESB,Server启动的时候抛了端口 ... -
好的算法
2010-11-11 20:00 758用数组代替Map---填入过程 public v ... -
Java调试参数
2010-09-28 08:48 3096java -Xdebug -Xrunjdwp:tran ... -
SQL语句
2010-09-26 13:31 691下列语句部分是Mssql语句,不可以在access中使用 ... -
使用Java JDK中Java.net包控制UDP协议
2010-09-26 12:38 878使用Java JDK中Java.net包控制UDP协议 什么 ... -
关于动态调用方法的例子
2010-09-26 12:37 719情况: 一个类中有很多同名方法,名字相同,参数列表不一样。 ...
相关推荐
JUnit 5 = JUnit Platform + JUnit Jupiter + JUnit Vintage,包含依赖包:junit-jupiter-5.5.1.jar,junit-jupiter-engine-5.5.1.jar,junit-jupiter-params-5.5.1.jar,junit-platform-launcher-1.5.1.jar,junit-...
JUnit是Java编程语言中最常用的单元测试框架之一,用于编写和运行可重复的自动化测试用例。Junit5.7.2版本是这个框架的一个稳定版本,提供了许多改进和新特性,使得测试更加高效且易于维护。这个离线jar文件包含了...
这个名为"junit5.jar"的文件正是JUnit 5的核心库,它包含了执行单元测试所需的所有核心组件,包括JUnit Platform、JUnit Jupiter和JUnit Vintage。本文将全面解析JUnit 5的关键特性和使用方法。 首先,JUnit ...
Files contained in junit4-4.8.2.jar: LICENSE.txt META-INF/MANIFEST.MF junit.extensions.ActiveTestSuite.class junit.extensions.RepeatedTest.class junit.extensions.TestDecorator.class junit.extensions...
JUnit是Java编程语言中最常用的单元测试框架之一,它允许开发者编写可重复运行的测试用例,以确保代码的正确性和稳定性。JUnit3.8.1是该框架的一个较早版本,尽管现在已经有更新的版本(如JUnit5),但了解其基本...
JUnit4测试框架是Java开发中广泛使用的单元测试工具,它为开发者提供了编写和运行可重复、可靠的测试用例的能力。这个“junit4测试jar包”包含了一切你需要在项目中集成JUnit4进行测试的库文件。只需将其复制到你的...
赠送jar包:junit-4.11.jar; 赠送原API文档:junit-4.11-javadoc.jar; 赠送源代码:junit-4.11-sources.jar; 赠送Maven依赖信息文件:junit-4.11.pom; 包含翻译后的API文档:junit-4.11-javadoc-API文档-中文...
赠送jar包:junit-4.12.jar; 赠送原API文档:junit-4.12-javadoc.jar; 赠送源代码:junit-4.12-sources.jar; 包含翻译后的API文档:junit-4.12-javadoc-API文档-中文(简体)版.zip 对应Maven信息:groupId:...
"JUnit in Action 3rd Edition" JUnit是一种流行的Java单元测试框架,由Kent Beck和Eric Gamma于1997年创立。JUnit在软件测试领域中扮演着重要的角色,帮助开发者编写高质量的代码。下面是关于JUnit的重要知识点: ...
JUnit是Java编程语言中最常用的单元测试框架之一,它允许开发者编写可执行的测试用例来验证代码的功能。这个“junit工具jar包”是JUnit 4.8.1版本,一个稳定且广泛使用的版本,提供了丰富的断言方法和测试注解,便于...
Junit5是Java开发中最常用的单元测试框架之一,它的出现为开发者提供了更加高效、灵活的测试体验。相较于之前的版本,Junit5引入了许多新的特性和改进,使得测试代码的编写和维护变得更加简单。本整合包包含了Junit5...
JUnit5相关的 JUnit Jupiter.jar依赖包,主要包含: junit-jupiter-5.5.1.jar,junit-jupiter-api-5.5.1.jar,junit-jupiter-engine-5.5.1.jar,junit-jupiter-params-5.5.1.jar,junit-jupiter-api-5.4.2.jar,...
赠送jar包:junit-4.13.2.jar; 赠送原API文档:junit-4.13.2-javadoc.jar; 赠送源代码:junit-4.13.2-sources.jar; 赠送Maven依赖信息文件:junit-4.13.2.pom; 包含翻译后的API文档:junit-4.13.2-javadoc-API...
JUnit4是Java编程语言中最广泛使用的单元测试框架之一,它为开发者提供了编写可重复执行、易于维护的测试代码的能力。这个“junit4 jar包”包含了运行JUnit4测试所必需的库文件,主要包括两个核心组件:`junit-4.11....
《JUnit in Action》是一本专为Java开发人员深入理解单元测试框架JUnit而编写的经典书籍。本书详尽地探讨了如何有效地使用JUnit进行软件测试,包括基础概念、高级特性和最佳实践,旨在帮助读者提升软件质量,降低...
JUnit5是Java编程语言中最流行的单元测试框架之一,它的最新版本带来了许多改进和新特性,使得测试更加高效和灵活。本资源包含的`junit5.jar`是JUnit5的运行库,可以用于运行使用JUnit5编写的测试用例。而`junit5-...
JUnit是Java编程语言中最常用的单元测试框架之一,它极大地简化了对代码进行验证和调试的过程。Junit4.12是该框架的一个稳定版本,它包含了许多改进和新特性,使得测试更加灵活和强大。在本文中,我们将深入探讨...
JUnit4是Java编程语言中最广泛使用的单元测试框架之一,它为开发者提供了一种方便、高效的方式来验证代码的正确性。这个“junit4 jar完整包”包含了所有你需要进行单元测试的类和接口,使得测试过程变得简单且易于...
JUnit5是Java编程语言中最广泛使用的单元测试框架之一,它为开发者提供了强大的测试能力,确保代码的质量和稳定性。本篇文章将详细介绍JUnit5的核心组件、主要功能以及如何在项目中使用。 JUnit5由三个主要模块组成...
### Junit单元测试内部机制深度解析 #### 一、自动化软件测试的重要性 随着软件工程的不断发展,软件的规模和复杂性急剧增加,软件测试成为确保软件质量和可靠性的关键环节。自动化测试,尤其是单元测试,因其高效...