在PHP中使用Mockery进行测试驱动开发(TDD)

WBOY
发布: 2016-06-20 12:55:35
原创
1742 人浏览过

测试驱动开发网上也谈了很多了,PHP方面的文章也有一些,在百度和Google里搜,好像没有看到几篇谈用Mock(伪装对象)的技术的,这里写篇文章讲讲。

先过一下测试驱动开发的基本理念:就是先写测试用例(一般这个测试用例都是自动化的单元测试用例,便于快速回滚执行),然后通过逐步修复测试用例的方法补齐产品代码,最后测试用例修复完毕后,产品也就写完了。

从我自己的实践中,我认为在类库开发的时候使用测试驱动开发技术是一个很好的方案,理由如下: 
能够写出测试用例,即说明对问题域已经有一个清晰的了解, 
节省了写文档的时间,测试用例就是类库调用的示例代码了。 
代码质量有保证,因为写类库的过程就是修复测试用例的过程,所以测试用例修复完毕后类库也就写完了。 
便于估时,估计类库开发时间的问题就简化成估计修复测试用例的时间了,相对于来说估时容易一些。

我们以编写一个字符串转数字的函数为例讲解测试驱动开发的理念,再引入Mock技术。在开始之前,需要安装PHPUnit和Mockery库(本文不使用PHPUnit自带的Mock库):

# 安装PHPUnitpear config-set auto_discover 1pear install pear.phpunit.de/PHPUnit# 安装Mock库sudo pear channel-discover pear.survivethedeepend.comsudo pear channel-discover hamcrest.googlecode.com/svn/pearsudo pear install --alldeps deepend/Mockery
登录后复制

那从测试驱动开发的理念来做的话,我们先写测试用例 ? parseinttest.php:

<?    class ParseIntTest extends PHPUnit_Framework_TestCase {        public function testParseIntBasic() {            $v = parse_int("12345");            $this->assertEquals(12345, $v);            $v = parse_int("-12345");            $this->assertEquals(-12345, $v);        /*            $v = parse_int("abcd");            $this->assertEquals(0, $v);            $v = parse_int("0xab12");            $this->assertEquals(0xab12, $v);            $v = parse_int("01b");            $this->assertEquals(1, $v);        */        }    }?>
登录后复制

上面的代码里,我们通过单元测试用例指定了要实现的函数parse_int的需求,即可以解析整数、带符号的整数等,又因为时间和资源的限制,那我们去掉了对十六进制和二进制的支持。

运行下面的命令执行测试用例:

phpunit --verbose parseinttest
登录后复制

因为这个时间没有任何代码,所以得到期望的错误 ? PHPUnit告诉我们找不到parse_int这个函数:
PHPUnit 3.6.12 by Sebastian Bergmann.

PHP Fatal error: Call to undefined function parse_int() in /var/www/pmdemo/parseinttest2.php on line 6

那我们先建一个空的parse_int函数 ? parseint.php:

<?    function parse_int($str) {        return 0;    }?>
登录后复制

上面的代码里我们先不实现函数parse_int的任何逻辑,再运行测试用例,得到:

shiyimin@ubuntu :/var/www/pmdemo$ phpunit --verbose parseinttest
PHPUnit 3.6.12 by Sebastian Bergmann.

F

Time: 0 seconds, Memory: 2.75Mb

There was 1 failure:

1) ParseIntTest::testParseIntBasic
Failed asserting that 0 matches expected 12345.

/var/www/pmdemo/parseinttest.php:7

FAILURES!
Tests: 1, Assertions: 1, Failures: 1.

从上面红色高亮显示出来的错误消息,我们知道是测试用例里的第一个判断没有通过,即下面的判断语句没有执行成功:

$this->assertEquals(12345, $v);
登录后复制

这是因为我们的parse_int函数总是返回0这个值,发现了这个错误,我们来补齐这个逻辑以修复测试用例:

parseint.php

<?    function parse_int($str) {        $result = 0;    $i = 0;        for ( $i = 0; $i < strlen($str); ++$i ) {            $result *= 10;        $result += $str[$i] - '0';    }    return $result;    }?>
登录后复制

再次运行测试用例:
shiyimin@ubuntu :/var/www/pmdemo$ phpunit --verbose parseinttest
PHPUnit 3.6.12 by Sebastian Bergmann.

F

Time: 0 seconds, Memory: 2.75Mb

There was 1 failure:

1) ParseIntTest::testParseIntBasic
Failed asserting that 12345 matches expected -12345.

/var/www/pmdemo/parseinttest.php:10

FAILURES!
Tests: 1, Assertions: 2, Failures: 1.

从上面的结果可以看出,第一个测试用例已经修复了,现在是在处理带符号的字符串时,发生了问题,继续修复代码:

parseint.php

<?    function parse_int($str) {        $result = 0;    $i = 0;    $neg = 1;                if ( $str[0] == '-' ) {         $neg = -1;    }    for ( $i = 0; $i < strlen($str); ++$i ) {            $result *= 10;        $result += $str[$i] - '0';    }    return $result * $neg;    }?>
登录后复制

再运行测试用例: 
shiyimin@ubuntu :/var/www/pmdemo$ phpunit --verbose parseinttest 
PHPUnit 3.6.12 by Sebastian Bergmann.

.

Time: 0 seconds, Memory: 2.75Mb

OK (1 test, 2 assertions)

好了,这次所有测试用例都通过了,那我们的产品代码的实现也告一段落了,当然文章为了讲解方便,上面的代码并不是一个完整的实现,例如它就无法处理字符串“+12345”的情形。

下篇文章讲解如何在PHPUnit里应用Mock技术。


来源:php.cn
本站声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责声明 Sitemap
PHP中文网:公益在线PHP培训,帮助PHP学习者快速成长!