九爷带你玩转 php单元测试

简介: 我总感觉 PHP 的开发者们并没有对 PHP 的质量有所追求,可能是因为 PHP 的机制问题吧,让大?糠值目⒄咦芤晕榔鞣梦示兔挥形侍猓院芏嗍焙颍? PHP 开发的,就没有单元测试的这些概念了。

<我不是九爷>

我总感觉 PHP 的开发者们并没有对 PHP 的质量有所追求,可能是因为 PHP 的机制问题吧,让大?糠值目⒄咦芤晕榔鞣梦示兔挥形侍猓院芏嗍焙颍? PHP 开发的,就没有单元测试的这些概念了。能不能有点追求?

我个人也是 PHP,但同时我也比较讨厌那些完事就算了的开发者,作为一个开发者,或者说是一个产品的经手人,就应该用心地去做好每个细节,一次比一次要更好。

但是做单元测试,质量检查,是需要一定的时间和人力投入的,但我敢保证地说,你花时间投入的,绝对不会是没用的,一定对你,对项目来说,是一个质的提升,只要你肯投入时间用心去做。

屁话说太多了,那接下来简单讲讲 phpunit 吧,官网

因为我们习惯用 composer,所以我们也使用 composer 安装吧。

wKiom1jk8HKDhRdpAAA9UQPaaZI550.jpg-wh_50

安装与配置

$ composer require phpunit/phpunit -vvv

安装完 phpunit,bin 执行脚本会创建在 vendor/bin 目录下,命名为 phpunit, 执行 php vendor/bin/phpunit 执行测试脚本

配置 bin 目录:

{
  "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>

可以通过配置目录和初始化信息,让脚本自动执行对应的测试用例。

基础使用

使用 PHPUnit 创建我们的测试用例:

<?phpclass 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 方法,代表类初始化,可以在初始化方法中初始化一些资源,或者加载。

数据库测试

除了以上基础的测试之外,关键一点应该在动态的数据,需要去测试吗,如果需要,那应该怎么去测试? 生产环境,也需要这样测试? 这个曾经困惑这我的问题,已经解开。

解答:

  1. composer 中,有 --no-dev 选项,用来部署生产环境,避免测试环境的数据或者代码跑在了生产环境下。并且生产环境上数据库操作是没有很高权限的操作,要是有的话,你得回去面壁思考一下了。

  2. dbunit 每次测试都重置数据,其实在生产环境下,就重置不了了,第一个是composer --no-dev 已经没有执行权利了,要是有,数据库已经不允许清空操作了。

  3. 要是生产环境不需要这些东西,那么应该怎么测试。其实需要有一个模拟生产环境的测试环境,去模拟生产环境测试,当所有测试都OK没有问题,那么就可以发布到生产环境上,要是严格一些,生产环境也是需要一轮测试。

$ composer require phpunit/dbunit -vvv

更多测试可看: 数据库测试

<?phpclass 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 提供了灵活的扩展操作(肯定啦,别人肯定不会像你这么傻,写死吧。哈哈),我们可以很容易地去实现自己的数据库测试类。

<?phpabstract 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;
    }
}

至今为止,完成了最基础和入门的单元测试和数据库测试,最终数据库无非就是查看数据增删改查是否和预期一样。所以,配置完数据库测试后,就可以走回第一步,编写你的测试用例,断言测试了。

恭喜你,你已经构建完自己的单元测试环境了。接下来需要做的是,提高易用性,测试覆盖率。我只能帮你到这里了,接下来的路,自己走吧。


目录
相关文章
|
测试技术 PHP
PHP: PhpStorm单元测试报错:PHPUnit Cannot open file
PHP: PhpStorm单元测试报错:PHPUnit Cannot open file
73 0
|
测试技术 PHP
phpunit PHP单元测试的利器
PHPUnit是PHP的单元测试框架。单元测试在软件开发中越来越受到重视,测试先行编程、极限编程和测试驱动开发在实践中被广泛。利用单元测试,也可以实现契约式设计。
117 0
phpunit PHP单元测试的利器
|
域名解析 前端开发 Java
使用 composer + phpstorm 为 PHP 单元测试助力
本文主要为大家讲解单元测试的相关内容,以及如何通过 composer + phpstorm 为 PHP 单元测试助力。
576 0
使用 composer + phpstorm 为 PHP 单元测试助力
|
测试技术 PHP
使用 composer + phpstorm 为 PHP 单元测试助力
PHP 单元测试的意义到底何在 我之前也有这样的困惑。看代码写的是否有问题,我们直接请求下就不 ok 了吗?为什么还要写单元测试呢?这不是多此一举吗? 那看我下面的场景举例吧。 单元测试一定要用框架吗 我觉得不用,直接一行脚本,可以不可以?我觉得 ok。
17982 0
|
测试技术 PHP 数据库
|
测试技术 PHP
PHP单元测试-保障代码正确运行的护卫神
代码能否正确运行是应用的关键,如何保证代码的运行就成为至关重要的事情,“对付坦克的最好工具就是另外一台坦克”,保证代码的正确运行就是使用代码来保证,软件的的需求是变动的,很难保证这一次的改动,会不会将引起上次需求的什么变化。如果是测试人员发现了可以修改,但是如果部署到线上,引起的问题就会严重的。 单元测试就是提供一些条件,这些条件将产生一定的结果,如果结果不符就不能通过测试,不能通过测试
1528 0
|
缓存 测试技术 PHP
PHP 单元测试问题记录
今天在mac上 使用单元测试时,测试本项目功能时 全部报错, 测试百度网页连接情况时,就正常 fox.风 经过各方实验,最后得出结果出现在权限问题上面(生成的缓存文件权限) 单元测试使用的当前用户组权限,PHP WEB项目使用的是php-fpm配置中的www用户组权限, 两者生成缓存文件权限不同,导致不能删除,一直报错 解决方法: 方式一:偷懒型 使用完单元测试
1254 0