单元测试(提升篇)
 

2008-12-25 作者:wukaichun600 来源:blogjava.net

 

本节是单元测试的第三篇。我以为这是重中之重的一章。单元测试的关键在于代码要可测。可测才能测。要做好单元测试,就必须在代码的可测性方面努力,在设计、重构方面用心。本篇主要分享我在如何写出可测性代码方面的理解,与大家共勉!

单元测试(提升篇)
             ------编写可测试性代码

一、可测试性设计

1. 接口依赖

这是最重要的一点,因为这可以使得我们很容易的针对一个接口实现Mock对象,模拟/替换实际对象会变的很容易。达到使一个被测对象处于一个孤立环境的测试要求。

这里,ClassA依赖于ClassB的具体实现,TestCase根本无法独立于ClassB对ClassA进行独立测试。

因此,我们将ClassA改为依赖于接口B_Inf。这样,可以很容易的实现一个Mock_B替换ClassB去ClassA进行孤立测试。

2. 依赖注入

一个方法对外部类的依赖,应该是可注入的。即可以通过构造方法、get/set方法的方式在外部将依赖关系注入。事实上,这也为我们在测试用例中替换待测类的依赖对象提供了机会。不应该出现在方法内部新建对象使用的情况。

3. 降低耦合度

待测类应与最少的类耦合,即最小交互原则。特别是要减少与那些离了具体环境就不能运行的类的耦合。可以通过门面模式等对外部调用进行隔离。

5.AOP

面向切面编程。给我们提供的启示是,将真正需要测的逻辑分离出来。摆脱那些无意义且简单重复的代码对测试的干扰。

6. 明确的契约

方法一定要有明确清晰的输入/输出。建议在方法的注释描述中,分三段“描述”“前置条件”“后置条件”。

二、可测试性重构

1. 可恶的静态方法

在我们的代码中有大量的调用静态方法的地方。用起来我们很爽,但这对测试来说却是灾难。因为我们除了通过改变代码创建stub来改变这些方法的行为外,我们没有任何途径。更要命的是,写这些stub的代价非常的巨大,常常令人望而却步。

解决方案:

将方法内部的这些调用提取成protected方法。在外部创建待测类的子类,重写该protected方法。

最佳实践:

这些静态方法由单态类提供,单态类由工厂方法获取,具体类使用这些单态类的接口。

我们在方法中通过接口使用对外部模块的调用。一方面,隔离了外部模块改变对我们产生的冲击,另一方面,也使我们使用Mock替换实际的外部组件,创建孤立测试环境成为可能。

2. 待测类方法中,new出另一个对象并使用其方法

两种方案:1)将new 出对象的过程封装成protected方法。同重构1。2)将该对象提取成类属性,即由使用关系变成关联关系。

3. 分离不可测/不必测代码

在不影响的情况下,将不可测部分分离到一些不需要测的简单方法中去。或者将可测的部分提取到一个私有方法中去。然后针对这个私有方法进行测试。

通常这种做法使用范围有限,但有些时候还是值的一试。

4. 单一职责

职责太多,肯定不好测。针对于这一点“不好测的方法必然不好用”。当方法过大,承担责任过多时,拆分是应该的。

5. 为类提供一个空构造方法。

我们的各个测试方法都依赖于对象处于一个特定的状态,满足一定的前置条件。而要将对象置为我们希望的状态,我们必须首先拥有一个对象。然而,很多时候,在一个隔离的单元测试环境下,构造函数由于各种原因不能正常初始化。

此时,可以为类提供一个的空的构造方法。在外部构造一个“裸”对象,而后根据前置条件将各个属性设置成需要的Mock对象。


火龙果软件/UML软件工程组织致力于提高您的软件工程实践能力,我们不断地吸取业界的宝贵经验,向您提供经过数百家企业验证的有效的工程技术实践经验,同时关注最新的理论进展,帮助您“领跑您所在行业的软件世界”。
资源网站: UML软件工程组织