本系列文章主要代码与文字来源于《单元测试的艺术》,原作者:Roy Osherove。译者:金迎。
本系列文章根据php的语法与使用习惯做了改编。所有代码在本机测试通过。如转载请注明出处。
上文介绍了通过创建一个局部的方法调用返回桩件,然后测试时用派生的子类来进行测试。
但是对于本文的示例来说,还有更简单的办法,不用桩件,也不用接口。
在被测类中,不是添加返回桩件的方法,而是添加直接返回计算结果的方法,然后在子类中覆盖这个方法,以此来回避对文件系统等外部依赖的调用。
因为这个方法最简单,所以可以优先考虑使用。
来对比一下上文的代码:需要去除接口,去除测试类中的接口实现。保留文件管理器类,保留被测类。修改被测类子类,修改被测类。
上文是源代码3个文件,测试代码3个文件。
现在是源代码2个文件,测试代码2个文件。
下面是全部代码:(所有的代码都有改动)
源代码
(1)t2\application\index\controller下文件管理器类,实现了上面的接口,但是实际被排除在单元测试之外,不测它。应该使用集成测试来测试此类。
FileExtensionManager.php
(2)t2\application\index\controller下被测类,日志分析器。使用了调用直接返回计算结果的方式来写代码,便于派生类覆盖,然后测试
LogAnalyzer.php
isValid($filename); } /** * @param string $filename * @return boolean */ protected function isValid($filename) { return (new FileExtensionManager())->isValid($filename); } }
测试代码
(3)t2\tests\index\controller\下,被测试类的子类,用于覆盖直接返回计算结果的方法,便于测试。因为这个子类测试专用,所以当然放在测试文件夹下。
LogAnalyzerExtend.php
isSupported; } }
(4)t2\tests\index\controller\下,最后是测试类,但不是测试被测试类,而是测试被测试类的子类。和上文不同的是,覆盖的是直接返回计算结果的方法。上文覆盖返回产生桩件的方法。
LogAnalyzerTest.php
isSupported = true; //调用并断言 $result = $analyzer->isValidLogFileName("short.ext"); $this->assertTrue($result); } }
cmd下测试通过。
上一篇: php单元测试进阶(10)- 核心技术 - 桩件(stub) - 调用方法注入桩件
下一篇: php单元测试进阶(12)- 核心技术 - mock对象