我总感受 PHP 的开发者们并无对 PHP 的质量有所追求,多是由于 PHP 的机制问题吧,让大部分的开发者总觉得浏览器访问就没有问题,因此不少时候,作 PHP 开发的,就没有单元测试的这些概念了。能不能有点追求?php
我我的也是 PHP,但同时我也比较讨厌那些完事就算了的开发者,做为一个开发者,或者说是一个产品的经手人,就应该用心地去作好每一个细节,一次比一次要更好。html
可是作单元测试,质量检查,是须要必定的时间和人力投入的,但我敢保证地说,你花时间投入的,绝对不会是没用的,必定对你,对项目来讲,是一个质的提高,只要你肯投入时间用心去作。mysql
屁话说太多了,那接下来简单讲讲 phpunit 吧,官网。sql
由于咱们习惯用 composer
,因此咱们也使用 composer
安装吧。数据库
$ composer require phpunit/phpunit -vvv
安装完 phpunit,bin 执行脚本会建立在 vendor/bin
目录下,命名为 phpunit
, 执行 php vendor/bin/phpunit
执行测试脚本json
配置 bin 目录:bootstrap
{ "config": { "bin": "bin" } }
配置 bin 目录产生的目录,执行 php bin/phpunit
脚本开始测试。数组
phpunit
能够配置在当前执行路径添加一个配置文件 phpunit.xml.dist
或者 phpunit.xml
,内容以下:浏览器
<phpunit colors="true" bootstrap="./vendor/autoload.php" > <testsuites> <testsuite> <directory>dir1</directory> </testsuite> <testsuite> <directory>dir2</directory> </testsuite> </testsuites> </phpunit>
能够经过配置目录和初始化信息,让脚本自动执行对应的测试用例。composer
使用 PHPUnit 建立咱们的测试用例:
<?php class DemoTest extends PHPUnit_Framework_TestCase { public function testPushAndPop() { $stack = []; $this->assertEquals(0, count($stack)); array_push($stack, 'foo'); $this->assertEquals('foo', $stack[count($stack)-1]); $this->assertEquals(1, count($stack)); $this->assertEquals('foo', array_pop($stack)); $this->assertEquals(0, count($stack)); } }
类名须要以 *Test 结尾,继承 PHPUnit_Framework_TestCase
。须要测试的方法须要一 test 开头,代表是一个测试方法。
通常经常使用测试无非就是 "断言",说白了,就是看看产生的结果是否是符合预期,若是是,那就证实,已经测试经过,不然,失败,说明逻辑处理,存在必定的差别,致使不符合预期。
更多的测试使用方法请看官网用例: PHPUnit
当咱们的测试对象继承了 PHPUnit 后,初始化方法就须要使用它自己提供的 setUp 方法,表明类初始化,能够在初始化方法中初始化一些资源,或者加载。
除了以上基础的测试以外,关键一点应该在动态的数据,须要去测试吗,若是须要,那应该怎么去测试? 生产环境,也须要这样测试? 这个曾经困惑这个人问题,已经解开。
解答:
composer 中,有 --no-dev 选项,用来部署生产环境,避免测试环境的数据或者代码跑在了生产环境下。而且生产环境上数据库操做是没有很高权限的操做,要是有的话,你得回去面壁思考一下了。
dbunit 每次测试都重置数据,其实在生产环境下,就重置不了了,第一个是composer --no-dev 已经没有执行权利了,要是有,数据库已经不容许清空操做了。
要是生产环境不须要这些东西,那么应该怎么测试。其实须要有一个模拟生产环境的测试环境,去模拟生产环境测试,当全部测试都OK没有问题,那么就能够发布到生产环境上,要是严格一些,生产环境也是须要一轮测试。
$ composer require phpunit/dbunit -vvv
更多测试可看: 数据库测试
<?php class DBTest extends PHPUnit_Extensions_Database_TestCase { /** * @return PHPUnit_Extensions_Database_DB_IDatabaseConnection */ public function getConnection() { $pdo = new PDO('mysql::dbname=test;host=127.0.0.1', 'user', 'pass'); return $this->createDefaultDBConnection($pdo, ':memory:'); } /** * @return PHPUnit_Extensions_Database_DataSet_IDataSet */ public function getDataSet() { return $this->createFlatXMLDataSet(dirname(__FILE__).'/_files/guestbook-seed.xml'); } }
getConnection
方法是获取数据库链接,继承数据库测试后,必须实现的一个方法,而且须要返回 PHPUnit_Extensions_Database_DB_IDatabaseConnection
对象,能够仿照上述写法便可。
getDataSet
方法是数据集,在建立数据库测试的时候,自动填充,测试,和删除。他执行的流程是,每一个测试用例,都会填充一次,以保证不会被其余测试用例影响。当当前测试用例测试完成后,会 truncate
掉填充的数据。
数据集支持挺多种方法,能够自定义数组,yml,xml,能够根据本身的使用习惯,自定义填充数据。数据集可看: 点我
执行脚本 php vendor/bin/phpunit
而后去对应查看本身的数据表,是否多了一些填充的数据呢?
在不少状况下,咱们的业务可谓是各类各样吧,假若 phpunit 提供的数据库测试还不能知足或者不够方便的时候,就须要扩展本身的数据库测试,来达到本身想要的效果。
幸亏,phpunit 提供了灵活的扩展操做(确定啦,别人确定不会像你这么傻,写死吧。哈哈),咱们能够很容易地去实现本身的数据库测试类。
<?php abstract class MyApp_Tests_DatabaseTestCase extends PHPUnit_Extensions_Database_TestCase { // 只实例化 pdo 一次,供测试的清理和装载基境使用 static private $pdo = null; // 对于每一个测试,只实例化 PHPUnit_Extensions_Database_DB_IDatabaseConnection 一次 private $conn = null; final public function getConnection() { if ($this->conn === null) { if (self::$pdo == null) { self::$pdo = new PDO('mysql::dbname=test;host=127.0.0.1', 'user', 'pass'); } $this->conn = $this->createDefaultDBConnection(self::$pdo, ':memory:'); } return $this->conn; } }
至今为止,完成了最基础和入门的单元测试和数据库测试,最终数据库无非就是查看数据增删改查是否和预期同样。因此,配置完数据库测试后,就能够走回第一步,编写你的测试用例,断言测试了。
恭喜你,你已经构建完本身的单元测试环境了。接下来须要作的是,提升易用性,测试覆盖率。我只能帮你到这里了,接下来的路,本身走吧。