我们将从构建简单的驼鹿监视软件开始示例各种场景下的测试驱动开发。
鹿是鹿家族中的最大个的成员。目前世界上估计有2百万头。为了更好的统计,世界驼鹿组织让我们开发一个驼鹿监视软件,希望保护人员能够跟踪和记录他们见到的驼鹿。
本文假设读者已经熟悉java/ant/maven/classpath,而且至少写过一两个简单的junit测试。
Vanilla JUnit
首先编写Moose类:
package moose;
import java.util.Date;
public class Moose {
private Date dateTimeObserved;
private int age;
public Moose(Date dateTimeObservedParam, int estimatedAge) {
this.dateTimeObserved = dateTimeObservedParam;
this.age = estimatedAge;
}
}
虽然驼鹿最长可以生存25年,但大部分驼鹿在年青的时候就被捕杀,通常只活了2、3年。根据世界驼鹿组织报告,我们想知道某一年龄后的驼鹿数据,因此我们增加了isOlderThan(int)方法。
单元测试如下:
public void testIsOlderThan() {
Moose moose = new Moose(new Date(), MOOSE_AGE);
assertTrue("Moose age "+ MOOSE_AGE +" should of been older than "+ TEST_AGE, moose.isOlderThan(TEST_AGE) );
}
方法:
boolean isOlderThan(int contenderAge) {
return this.age >= contenderAge;
}
这是一个普通的junit测试用例,这种方式通常用在测试驱动的开发中。这种完全独立的方法一般是很少的;通常我们都需要一些代价昂贵的或比较难构建的外部资源。
使用伪对象和jMock
观察驼鹿的最佳时间是在黎明和黄昏,跟其他动物的观察时间相似。WOM希望我们提供获取记录驼鹿被发现的时间的保护人员,这样就可以通过String getObserverName()来获取相应的驼鹿了。
不幸地是,保护人员的接口来自第三方接口PersonnelUnit,而这是个大家伙没有LDAP我们是无法构造的。
为了测试getObserverName(),我们可以启动LDAP服务器,发送数据,运行测试用例,然后再关闭LDAP。但那将是集成测试,而不是单元测试了,对我们来说宁可多写些代码。因此,我们创建了一个包含我们需要方法的Ranger接口:
public interface Ranger {
String getName();
}
这给我们第一个启发式的单元测试:用接口分离外部依赖。
我们需要改变Moose的构造函数增加Ranger参数。于是变成:
... above as before...
private Ranger observer;
public Moose( Date dateTimeObservedParam,
int estimatedAge,
Ranger observedBy)
{
this.dateTimeObserved = dateTimeObservedParam;
this.age = estimatedAge;
this.observer = observedBy;
}
... below as before ...