第二节 PHPUnit测试的剖析

       现在,让我们仔细看看测试结构的样子。 让我们从一个简单的测试用例开始,它将显示基本的PHPUnit测试结构。 以下代码片段是测试用于排序数组的两个PHP函数的一个非常基本的示例:asort()用于对数组进行排序并维护索引,而ksort()用于按键对数组进行排序。 首先,我们有一系列蔬菜,其中名称是关键,价格是价值:

<?php

class SecondTest extends PHPUnit_Framework_TestCase
{
    public function testAsort ()
    {
        $vegetablesArray = array (
            'carrot' => 1, 'broccoli' => 2.99,
            'garlic' => 3.98, 'swede' => 1.75
        );
        $sortedArray     = array (
            'carrot'   => 1, 'swede' => 1.75,
            'broccoli' => 2.99, 'garlic' => 3.98
        );
        asort( $vegetablesArray, SORT_NUMERIC );
        $this->assertSame( $sortedArray, $vegetablesArray );
    }

    public function testKsort ()
    {
        $fruitsArray = array (
            'oranges' => 1.75, 'apples' => 2.05,
            'bananas' => 0.68, 'pear' => 2.75
        );
        $sortedArray = array (
            'apples'  => 2.05, 'bananas' => 0.68,
            'oranges' => 1.75, 'pear' => 2.75
        );
        ksort( $fruitsArray, SORT_STRING );
        $this->assertSame( $sortedArray, $fruitsArray );
    }
}

       如您所见,每个测试用例都是一个类。 类名应以Test结尾。 这是因为当您测试类时,测试类应该镜像测试类,并且测试类和测试套件之间的区别是测试添加到名称:

.
|-- src
    '-- library
        '-- Util
            '-- File.php
|-- tests
    '-- library
        '-- Util
            '-- FileTest.php

       每个测试类也称为测试套件,它扩展了PHPUnit_Framework_TestCase类。 当然,您可以编写自己的修改或扩展的PHPUnit_ Framework_TestCase并调用它,例如,MyTestCase,但这个父类应该扩展原始的PHPUnit类。 这可以在以下类定义中看到:

abstract class PHPUnit_Framework_TestCase extends PHPUnit_Framework_Assert implements PHPUnit_Framework_Test,PHPUnit_Framework_SelfDescribing

       以下内容告诉您前面的类定义的作用:

  • abstract:这是因为你总是扩展PHPUnit_Framework_TestCase类而不直接执行这个类
  • 扩展PHPUnit_Framework_Assert:这是提供一组断言方法的类,例如assertTrue()和assertEquals()
  • 实现PHPUnit_Framework_Test:这是一个简单的接口,只包含一个方法run()来执行测试
  • 实现PHPUnit_Framework_SelfDescribing:这是另一个简单的接口,只包含一个方法toString()

       PHPUnit不使用已经引入PHP 5.3的名称空间。 主要是为了向后兼容,PHPUnit在类名中使用下划线来匹配文件系统中的类位置。

       1.定义测试方法

       每个测试方法名称都应该以test开头。 那么这个名字取决于你。 您可以镜像他们的测试类方法,但您可能需要更多测试。 每个测试方法都应该有一个解释性名称,这意味着您应该能够从名称中说出您正在测试的内容,例如testChangePassword()或testChangePasswordForLockedAcoounts()

       2.测试函数

       这是一个理论,现在我们来看看一个特定的现实生活中的例子。 让我们使用开发人员在求职面试中可以得到的以下问题,并通过此示例了解PHPUnit如何帮助我们解决问题或让我们确信我们找到了正确的解决方案。

       任务是在PHP中编写一个函数,它返回有序数组中最大的连续整数和。

       例如,如果输入为[0,1,2,3,6,7,8,9,11,12,14],则最大总和为6 + 7 + 8 + 9 = 30。

       如您所见,它看起来并不特别困难。 您可以采取几种不同的方法。 写一些非常聪明的东西,使用蛮力和遍历数组,但任何有效的方法都是正确的答案。

       现在,让我们来看看PHPUnit如何帮助解决这个问题。 首先,你必须停下来思考你在做什么 - 不仅仅是开始编写代码,并且在一段时间后迷失方向,不知道你想要实现什么。

       那么你如何解决这个问题呢?

       以下步骤是最好的方法:

               1.使用PHP函数usort()使用用户定义的比较函数按值对数组进行排序。

                2.然后返回已排序数组的最后一个元素,该元素将成为最大的组。

       这是制定战略的重要一点。 现在,让我们将其转换为函数,而无需编写实现。 请考虑以下代码段:

<?php
/**
* 返回连续整数的最大总和
* @param array $inputArray
* @return array
*/
function sumFinder(array $inputArray) : array {}
/**
 * 自定义数组比较方法
 * @param array $a
 * @param array $b
 * @return int
 */
function compareArrays(array $a, array $b) : int {}

        现在你有两个功能。 第一个函数使用PHP函数usort()将数组作为输入,第二个函数compareArrays()按组对数组进行排序。 然后第一个函数返回一个包含组名和sum的数组。

        现在你知道函数的样子了。 让我们为它们编写测试,描述它们的行为方式,如下面的代码片段所示:

<?php
require_once 'SumFinder.php';

class SumFinderTest extends PHPUnit_Framework_TestCase
{
    public function testSumFinder ()
    {
        $input  = array ( 0, 1, 2, 3, 6, 7, 8, 9, 11, 12, 14 );
        $result = array ( 'group' => '6, 7, 8, 9', 'sum' => 30 );
        $this->assertEquals( $result, sumFinder( $input ) );
    }

    public function testCompareArrays ()
    {
        $array1 = array ( 0, 1, 2, 3 );
        $array2 = array ( 6, 7, 8, 9 );
// $array2 > $array1
        $this->assertEquals( -1, compareArrays( $array1, $array2 ) );
// $array1 < $array2
        $this->assertEquals( 1, compareArrays( $array2, $array1 ) );
// $array2 = $array2
        $this->assertEquals( 0, compareArrays( $array2, $array2 ) );
    }
}

       如您所见,每个函数都有两个简单的测试。 第一个测试testSumFinder()验证主sumFinder()函数,第二个测试testCompareArrays()验证在使用usort()时数组是否正确排序。 现在您可以运行它们并查看以下结果:

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

      牛逼! 这就是我们想要的:它失败了,因为我们必须编写一个实现。让我们从compareArrays()函数开始,如下面的代码片段所示:

<?php
/**
 * 自定义数组比较方法
 * @param array $a
 * @param array $b
 * @return int
 */
function compareArrays ( array $a, array $b )
{
    $sumA = array_sum( $a );
    $sumB = array_sum( $b );
    if ($sumA == $sumB) {
        return 0;
    } else if ($sumA > $sumB) {
        return 1;
    } else {
        return -1;
    }
}

       然后你可以运行testCompareArrays(),很好,它的工作方式如下面的输出:

OK (1 test, 3 assertions)

  现在我们可以转到sumFinder()实现,如下面的代码片段所示:

<?php
/**
 * 返回连续整数的最大总和
 * @param array $inputArray
 * @return array
 */

function sumFinder ( array $inputArray ) : array
{
    $arrayGroups = array ();
    foreach ($inputArray as $element) {
//initial settings
        if (!isset( $previousElement )) {
            $previousElement  = $element;
            $arrayGroupNumber = 0;
        }
        if (( $previousElement + 1 ) != $element) {
            $arrayGroupNumber += 1;
        }
        $arrayGroups[ $arrayGroupNumber ][] = $element;
        $previousElement                    = $element;
    }
    usort( $arrayGroups, 'compareArrays' );
    $highestGroup = array_pop( $arrayGroups );
    return ( array (
        'group'                    => implode( ', ',
            $highestGroup ), 'sum' => array_sum( $highestGroup )
    ) );
}

       运行上面的代码段时,您将获得以下结果:

OK (1 test, 1 assertion)

  这就是解决方案。 在这里,您应该看到单元测试的最大优势之一; 它会强迫您在开始编写代码或在实现之前阐明代码,考虑您正在做什么并有一个清晰的想法。

       3.测试方法

       测试方法与测试函数非常相似,但是你可以做更多。过程编程没有任何问题,但是有很好的理由使用面向对象编程(OOP)。 PHP严格地说不是Java或C#等OOP语言,但这并不意味着你不能使用OOP。 当然可以,PHP有一个非常好的对象模型。 它是一个可靠的实现,具有您期望的所有主要功能。 OOP真正有用的地方在于组织具有所有OOP优势的代码,并且在编写测试时也有帮助。

       PHPUnit支持测试类和对象。 在测试类时,PHPUnit和单元测试的真正优势在于您不仅可以测试每个方法,而且在必要时,您可以使用您的实现替换部分类以进行测试,例如,避免将数据存储在数据库中。 这称为测试双打,这些技术将在本书的后面部分进行描述。

       在测试类方法时,建议仅测试公共方法。 据说尽可能多的代码应该进行测试后,这听起来有点奇怪。 为什么不为私有或受保护的方法编写测试? 原因是你应该只测试公共方法; 其他一切都只是你不应该担心的内部实现,即使它发生了变化,公共方法仍然应该以相同的方式运行。 如果私有/受保护的方法过于复杂,则可能表明它们应该拥有自己的类,或者您可以使用PHP反射并动态更改可访问性。

   代码实现部分待续......

猜你喜欢

转载自www.cnblogs.com/mysic/p/9440511.html