模拟对象(mock)是真实对象的替代物,用来测试真实对象的部分行为。如果你看过电影《外星人入侵》(Body Snatchers),应该已经对模拟对象有基本的了解了。一般来说,只有在不方便创建真实对象(例如数据库连接)时,或者对真实对象的测试会产生不希望有的副作用时,才需要用到模拟对象。例如,我们可能不希望在测试过程中,把数据写入到文件系统或数据库中。
本章我们将测试一个核反应堆——当然不是真的,它只是一个类。这个核反应堆类可以做阶乘的计算。假设求阶乘会引起一系列的反应,导致发生核灾难的后果。我们将使用mock包,用一个模拟对象来模拟阶乘计算。
具体步骤
首先我们将安装mock包,然后创建一个模拟对象并用它来测试一段代码。
- 安装mock。
为了安装mock包,请执行如下的命令。
- 创建一个模拟对象。
核反应堆类有一个do_work
方法,该方法会调用危险的factorial
方法。我们希望用模拟对象替代factorial
方法。用如下方式创建模拟对象。
这将确保模拟对象被调用后,其返回值是6。
- 断言行为。
我们可以用若干种方法,检查模拟对象的行为,进而测试真实对象的行为。例如,我们可以断言,我们用正确的参数调用了潜在的有爆炸危险的factorial
方法,具体如下所示。
使用了模拟对象的完整的测试代码如下。
我们向factorial
方法传递了一个字符串,目的是说明模拟对象并不执行真实对象中的代码。单元测试的作用和上一攻略中介绍的相同。第二个测试没有对核反应堆类做任何测试,只是为了演示不使用模拟对象而直接执行真实代码的情况。
测试后的输出结果如下。
攻略小结
模拟对象不具有任何真实的行为。他们就像是伪装成人类的外星人,但还达不到外星人的智慧程度。正如外星人不能说出他所替换的真实人的生日一样,我们需要对模拟对象进行设置,使其能以适当的方式作出反应。例如,本例中的模拟对象返回6。我们可以记录模拟对象的调用情况——被调用了多少次,用的什么参数。