PHP单元测试框架PHPUnit的使用

之前在学习IOS开发时有专门写过Objective-C的单元测试的文章,IOS开发学习之单元测试,今天再总结下怎么在PHP中使用单元测试。php

1、前言

在这篇文章中,咱们使用 composer 的依赖包管理工具进行phpunit包安装和管理,composer 官方地址 https://getcomposer.org/,按照提示进行全局安装便可,另外,咱们也会使用一个很是好用的Monolog记录日志组件记录日志,方便咱们查看。html

在根目录下创建 coomposer.json 的配置文件,输入如下内容:git

{
    "autoload": {
        "classmap": [
            "./"
        ]
    }
}

上面的意思是将根目录下的全部的类文件都加载进来, 在命令行执行 composer install 后,在根目录会生成出一个vendor的文件夹,咱们之后经过 composer 安装的任何第三方代码都会被生成在这里。github

2、为何要单元测试?

只要你想到输入一些东西到print语句或调试表达式中,就用测试代替它。 --Martin Fowlerweb

PHPUnit 是一个用PHP编程语言开发的开源软件,是一个单元测试框架。PHPUnit由Sebastian Bergmann建立,源于Kent Beck的SUnit,是xUnit家族的框架之一。编程

单元测试是对单独的代码对象进行测试的过程,好比对函数、类、方法进行测试。单元测试可使用任意一段已经写好的测试代码,也可使用一些已经存在的测试框架,好比JUnit、PHPUnit或者Cantata++,单元测试框架提供了一系列共同、有用的功能来帮助人们编写自动化的检测单元,例如检查一个实际的值是否符合咱们指望的值的断言。单元测试框架常常会包含每一个测试的报告,以及给出你已经覆盖到的代码覆盖率。json

总之一句话,使用 phpunit 进行自动测试,会使你的代码更健壮,减小后期维护的成本,也是一种比较标准的规范,现现在流行的PHP框架都带了单元测试,如Laraval,Symfony,Yii2等,单元测试已经成了标配。segmentfault

另外,单元测试用例是经过命令操控测试脚本的,而不是经过浏览器访问URL的。浏览器

3、安装PHPUnit

使用 composer 方式安装 PHPUnit,其余安装方式请看这里app

composer require --dev phpunit/phpunit ^6.2

安装 Monolog 日志包,作 phpunit 测试记录日志用。

composer require monolog/monolog

安装好以后,咱们能够看coomposer.json 文件已经有这两个扩展包了:

"require": {  
     "monolog/monolog": "^1.23",
    },

 "require-dev": {
        "phpunit/phpunit": "^6.2"
    },

4、PHPUnit简单用法

一、单个文件测试

建立目录tests,新建文件 StackTest.php,编辑以下:

<?php
/**
 * 一、composer 安装Monolog日志扩展,安装phpunit单元测试扩展包
 * 二、引入autoload.php文件
 * 三、测试案例
 *
 *
 */
namespace App\tests;
require_once __DIR__ . '/../vendor/autoload.php';
define("ROOT_PATH", dirname(__DIR__) . "/");

use Monolog\Logger;
use Monolog\Handler\StreamHandler;

use PHPUnit\Framework\TestCase;


class StackTest extends TestCase
{
    public function testPushAndPop()
    {
        $stack = [];
        $this->assertEquals(0, count($stack));

        array_push($stack, 'foo');

        // 添加日志文件,若是没有安装monolog,则有关monolog的代码均可以注释掉
        $this->Log()->error('hello', $stack);

        $this->assertEquals('foo', $stack[count($stack)-1]);
        $this->assertEquals(1, count($stack));

        $this->assertEquals('foo', array_pop($stack));
        $this->assertEquals(0, count($stack));
    }

    public function Log()
    {
        // create a log channel
        $log = new Logger('Tester');
        $log->pushHandler(new StreamHandler(ROOT_PATH . 'storage/logs/app.log', Logger::WARNING));
        $log->error("Error");
        return $log;
    }
}

代码解释:

  1. StackTest为测试类
  2. StackTest 继承于 PHPUnit\Framework\TestCase
  3. 测试方法testPushAndPop(),测试方法必须为public权限,通常以test开头,或者你也能够选择给其加注释@test来表
  4. 在测试方法内,相似于 assertEquals() 这样的断言方法用来对实际值与预期值的匹配作出断言。

命令行执行:
phpunit 命令 测试文件命名

➜  framework#  ./vendor/bin/phpunit tests/StackTest.php

// 或者能够省略文件后缀名
//  ./vendor/bin/phpunit tests/StackTest

执行结果:

➜  framework# ./vendor/bin/phpunit tests/StackTest.php
PHPUnit 6.4.1 by Sebastian Bergmann and contributors.

.                                                                   1 / 1 (100%)

Time: 56 ms, Memory: 4.00MB

OK (1 test, 5 assertions)

咱们能够在app.log文件中查看咱们打印的日志信息。

二、类文件引入

Calculator.php

<?php  
class Calculator  
{  
    public function sum($a, $b)  
    {  
        return $a + $b;  
    }  
}  
?>

单元测试类:
CalculatorTest.php

<?php

namespace App\tests;
require_once __DIR__ . '/../vendor/autoload.php';
require "Calculator.php";

use PHPUnit\Framework\TestCase;


class CalculatorTest extends TestCase
{
    public function testSum()
    {
        $obj = new Calculator;
        $this->assertEquals(0, $obj->sum(0, 0));

    }

}

命令执行:

> ./vendor/bin/phpunit tests/CalculatorTest

执行结果:

PHPUnit 6.4.1 by Sebastian Bergmann and contributors.

F                                                                   1 / 1 (100%)

Time: 117 ms, Memory: 4.00MB

There was 1 failure:

若是咱们把这里的断言故意写错,$this->assertEquals(1, $obj->sum(0, 0));
看执行结果:

PHPUnit 6.4.1 by Sebastian Bergmann and contributors.

F                                                                   1 / 1 (100%)

Time: 117 ms, Memory: 4.00MB

There was 1 failure:

1) App\tests\CalculatorTest::testSum
Failed asserting that 0 matches expected 1.

/Applications/XAMPP/xamppfiles/htdocs/web/framework/tests/CalculatorTest.php:22

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

会直接报出方法错误信息及行号,有助于咱们快速找出bug

三、高级用法

你是否已经厌烦了在每个测试方法命名前面加一个test,是否由于只是调用的参数不一样,却要写多个测试用例而纠结?我最喜欢的高级功能,如今隆重推荐给你,叫作框架生成器

Calculator.php

<?php  
class Calculator  
{  
    public function sum($a, $b)  
    {  
        return $a + $b;  
    }  
}  
?>

命令行启动测试用例,使用关键字 --skeleton

> ./vendor/bin/phpunit --skeleton Calculator.php

执行结果:

PHPUnit 6.4.1 by Sebastian Bergmann and contributors.

Wrote test class skeleton for Calculator to CalculatorTest.php.

是否是很简单,由于没有测试数据,因此这里加测试数据,而后从新执行上边的命令

<?php  
class Calculator  
{  
    /** 
     * @assert (0, 0) == 0 
     * @assert (0, 1) == 1 
     * @assert (1, 0) == 1 
     * @assert (1, 1) == 2 
     */  
    public function sum($a, $b)  
    {  
        return $a + $b;  
    }  
}  
?>

原始类中的每一个方法都进行@assert注解的检测。这些被转变为测试代码,像这样

/**
     * Generated from @assert (0, 0) == 0.
     */
    public function testSum() {
        $obj = new Calculator;
        $this->assertEquals(0, $obj->sum(0, 0));
    }

执行结果:

./vendor/bin/phpunit tests/CalculatorTest
PHPUnit 6.4.1 by Sebastian Bergmann and contributors. 
  
....  
  
Time: 0 seconds  
  
  
OK (4 tests)

四、其余用法

其余用法请参考官网:PHPUnit中国官网


参考文章:
PHPUnit中国官网文档

相关文章
相关标签/搜索