在上篇文章中介绍了如何使用PHPUnit进行单元测试,现在我就来谈谈如何编写测试用例以及怎样保证测试的全面性。

通常的测试用例继承自 PHPUnit_Framework_TestCase类,其中的每个测试都以test开头,而且声明为公共类型public。每个测试用例都有一个构建 方法setUp()和拆除方法tearDown(),分别在每个测试执行之前和之后执行,这两个方法都声明为被保护类型 protected。测试语句的类型包括断言、标记跳过、标记未完成。自动生成的测试类使用标记未完成来表示该测试未完成,在测试条件不满足的情况下要使 用标记跳过,如测试Oracle数据库驱动时没有Oracle数据库环境、Linux下无法测试SQL Server数据库驱动等。测试结果包括成功、失败和错误。出现错误的结果说明你的代码中有语法或运行时错误,这些错误要首先被解决。
[separator]
标记未完成
在开始写测试用例时,我们使用标记跳过来表示测试是未完成的,这样做和什么都不写的区别是后者会认为测试是成功的,而你可能会在之后忘了写测试。
PHP代码
   1. 
   2. class UnitTest extends PHPUnit_Framework_TestCase 
   3. { 
   4.     /**
   5.      * 测试Hello()方法
   6.      */ 
   7.     public function testHello() 
   8.     { 
   9.         $this->markTestIncomplete('这是一个未完成的测试'); 
  10.     } 
  11. } 
  12. ?> 
标记跳过
使用标记跳过来跳过不满足测试条件的情况,避免出现错误而影响测试结果。
PHP代码
   1. 
   2. class UnitTest extends PHPUnit_Framework_TestCase 
   3. { 
   4.     protected function setUp() 
   5.     { 
   6.         // 该测试用例需要xxx扩展,如果不满足就跳过 
   7.         if (!extension_loaded('xxx')) { 
   8.             $this->markTestSkipped('这是一个跳过的测试'); 
   9.         } 
  10.     } 
  11. } 
  12. ?> 
断言
布尔类型
assertTrue   断言为真
assertFalse  断言为假
NULL类型
assertNull     断言为NULL
assertNotNull  断言非NULL
数字类型
assertEquals             断言等于
assertNotEquals          断言不等于
assertGreaterThan        断言大于
assertGreaterThanOrEqual 断言大于等于
assertLessThan           断言小于
assertLessThanOrEqual    断言小于等于
字符类型
assertEquals          断言等于
assertNotEquals       断言不等于
assertContains        断言包含
assertNotContains     断言不包含
assertContainsOnly    断言只包含
assertNotContainsOnly 断言不只包含
数组类型
assertEquals          断言等于
assertNotEquals       断言不等于
assertArrayHasKey     断言有键
assertArrayNotHasKey  断言没有键
assertContains        断言包含
assertNotContains     断言不包含
assertContainsOnly    断言只包含
assertNotContainsOnly 断言不只包含
对象类型
assertAttributeContains           断言属性包含
assertAttributeContainsOnly       断言属性只包含
assertAttributeEquals             断言属性等于
assertAttributeGreaterThan        断言属性大于
assertAttributeGreaterThanOrEqual 断言属性大于等于
assertAttributeLessThan           断言属性小于
assertAttributeLessThanOrEqual    断言属性小于等于
assertAttributeNotContains        断言不包含
assertAttributeNotContainsOnly    断言属性不只包含
assertAttributeNotEquals          断言属性不等于
assertAttributeNotSame            断言属性不相同
assertAttributeSame               断言属性相同
assertSame                        断言类型和值都相同
assertNotSame                     断言类型或值不相同
assertObjectHasAttribute          断言对象有某属性
assertObjectNotHasAttribute       断言对象没有某属性
class类型
class类型包含对象类型的所有断言,还有
assertClassHasAttribute          断言类有某属性
assertClassHasStaticAttribute    断言类有某静态属性
assertClassNotHasAttribute       断言类没有某属性
assertClassNotHasStaticAttribute 断言类没有某静态属性
文件相关
assertFileEquals     断言文件内容等于
assertFileExists     断言文件存在
assertFileNotEquals  断言文件内容不等于
assertFileNotExists  断言文件不存在
XML相关
assertXmlFileEqualsXmlFile        断言XML文件内容相等
assertXmlFileNotEqualsXmlFile     断言XML文件内容不相等
assertXmlStringEqualsXmlFile      断言XML字符串等于XML文件内容
assertXmlStringEqualsXmlString    断言XML字符串相等
assertXmlStringNotEqualsXmlFile   断言XML字符串不等于XML文件内容
assertXmlStringNotEqualsXmlString 断言XML字符串不相等
有返回值的方法或函数根据其类型选择相应的断言,下面是一个简单例子。
PHP代码
   1. 
   2. class UnitTest extends PHPUnit_Framework_TestCase 
   3. { 
   4.     /**
   5.      * 测试返回值为布尔类型
   6.      */ 
   7.     public function testReturnBool() 
   8.     { 
   9.         // 实际情况把TRUE和FALSE换为被测试方法或函数 
  10.         $this->assertTrue(TRUE); 
  11.         $this->assertFalse(FALSE); 
  12.     } 
  13.     /**
  14.      * 测试返回值为字符串类型
  15.      */ 
  16.     public function testReturnString() 
  17.     { 
  18.         $expected = 'string'; 
  19.         // 实际情况把下面的'string'换为被测试方法或函数 
  20.         $result = 'string'; 
  21.         $this->assertEquals($expected, $result); 
  22.     } 
  23.     /**
  24.      * 测试返回值是数字类型
  25.      */ 
  26.     public function testReturnInt() 
  27.     { 
  28.         $expected = 10; 
  29.         // 实际情况把20换为被测试方法或函数 
  30.         $result = 20; 
  31.         $this->assertGreaterThan($expected, $result); 
  32.     } 
  33.     /**
  34.      * 测试返回值是数组类型
  35.      */ 
  36.     public function testReturnArray() 
  37.     { 
  38.         // 实际情况把$result赋值为被测试方法或函数 
  39.         $result = array('test' => 'hello'); 
  40.         // 实际情况把'test'换为要测试的键名称 
  41.         $this->assertArrayHasKey('test', $result); 
  42.     } 
  43.     /**
  44.      * 测试返回值是对象类型
  45.      */ 
  46.     public function testReturnObject() 
  47.     { 
  48.         // 实际情况把$this换为期望的对象 
  49.         $expected = $this; 
  50.         // 实际情况把$this换为被测试方法或函数 
  51.         $result = $this; 
  52.         $this->assertSame($expected, $result); 
  53.     } 
  54. } 
  55. ?> 
无返回值的方法,可以通过其他方法读取属性,也可以使用对象类型中的断言来判断属性的改变。
PHP代码
   1. 
   2. /**
   3.  * Unit类,有一个无返回值方法
   4.  */ 
   5. class Unit 
   6. { 
   7.     protected $name; 
   8.     /**
   9.      * 设置name属性
  10.      */ 
  11.     public function setName($value) 
  12.     { 
  13.         $this->name = $value; 
  14.     } 
  15. } 
  16. ?> 
PHP代码
   1. 
   2. require_once 'Unit.php'; 
   3.  
   4. class UnitTest extends PHPUnit_Framework_TestCase 
   5. { 
   6.     /**
   7.      * 测试无返回值的方法
   8.      */ 
   9.     public function testsetName() 
  10.     { 
  11.         $expected = 'Hello'; 
  12.         $o = new Unit(); 
  13.         $o->setName('Hello'); 
  14.         $this->assertAttributeEquals($expected, 'name', $o); 
  15.     } 
  16. } 
  17. ?> 
编写测试
编写测试的原则是,尽可能测试每种不同的参数调用和不同的返回结果类型,既要测试成功的情况,也要测试失败的情况;无返回值的情况,要测试属性改变、输出内容、异常类型等;测试后记得要恢复现场。
在这里做了超出自己能力的事并不光荣。在你写某个函数之前,你只想让它做加法,但它却能做乘法,而且单元测试正确通过。我们来看看它是怎么做到的。
PHP代码
   1. 
   2. /**
   3.  * 计算器类
   4.  */ 
   5. class Calculator 
   6. { 
   7.     /**
   8.      * 做加法运算
   9.      *
  10.      * @param  int  $a
  11.      * @param  int  $b
  12.      * @return int
  13.      */ 
  14.     public function add($a, $b) 
  15.     { 
  16.         return $a * $b; 
  17.     } 
  18. } 
  19. ?>
PHP代码
   1. 
   2. require_once 'Calculator.php'; 
   3.  
   4. class CalcuatorTest extends PHPUnit_Framework_TestCase 
   5. { 
   6.     public function testadd() 
   7.     { 
   8.         // 创建实例 
   9.         $c = new Calculator(); 
  10.         $expected = 4; 
  11.         // 做加法 
  12.         $result = $c->add(2, 2); 
  13.         $this->assertEquals($expected, $result); 
  14.     } 
  15. } 
  16. ?> 
合理的测试能帮助我们尽早发现错误。add()方法有两个参数,测试的时候用了两个值相同的参数。如果多做几次测试又太麻烦,对于这个测试我们按照科学的方法只需要一次。从概率学上讲,当你使用的参数差异越大时,结果相同的概率越低。
对于只有几个返回值的情况,要测试全部,如布尔类型。
PHP代码
   1. 
   2. /**
   3.  * File.php
   4.  */ 
   5.  
   6. /**
   7.  * 在文件中写入数据并保存
   8.  *
   9.  * @param  string  $path
  10.  * @param  string  $data
  11.  * @return bool
  12.  */ 
  13. function Save($path, $data) 
  14. { 
  15.     if (is_dir($path)) { 
  16.       return FALSE; 
  17.     } 
  18.  
  19.     return file_put_contents($path, $data); 
  20. } 
  21. ?>
PHP代码
   1. 
   2. /**
   3.  * FileTest.php
   4.  */ 
   5.  
   6. require_once 'File.php'; 
   7.  
   8. class FileTest extends PHPUnit_Framework_TestCase 
   9. { 
  10.     /**
  11.      * 测试保存文件
  12.      */ 
  13.     public function testSave() 
  14.     { 
  15.         $file = 'IamFile.txt'; 
  16.         $dir = 'IamDir'; 
  17.         mkdir($dir); 
  18.  
  19.         // 测试返回值为真的情况 
  20.         $this->assertTrue(Save($file, 'TestTrue')); 
  21.  
  22.         // 测试返回值为假的情况 
  23.         $this->assertFalse(Save($dir, 'TestFalse')); 
  24.  
  25.         // 恢复现场 
  26.         if (is_file($file)) { 
  27.           unlink($file); 
  28.         } 
  29.         rmdir($dir); 
  30.     } 
  31. } 
  32. ?> 
对于有多种类型返回值或不同参数的情况,分别测试每种类型和参数。下面是ThinkPHP源代码中的一个函数,有点复杂。这个例子不能单独运行,如需要请用SVN导出最新的ThinkPHP源代码(含单元测试)。
PHP代码
   1. 
   2. /**
   3.  * URL生成函数
   4.  *
   5.  * @param  string  $action  方法名
   6.  * @param  string  $module  模块名
   7.  * @param  string  $route   路由名
   8.  * @param  array   $params  参数
   9.  */ 
  10. function url($action=ACTION_NAME,$module=MODULE_NAME,$route='',$app=APP_NAME,$params=array()) 
  11. { 
  12.     if(C('DISPATCH_ON') && C('URL_MODEL')>0) { 
  13.         switch(C('PATH_MODEL')) { 
  14.             case 1:// 普通PATHINFO模式 
  15.                 $str    =   '/'; 
  16.                 foreach ($params as $var=>$val) 
  17.                     $str .= $var.'/'.$val.'/'; 
  18.                 $str = substr($str,0,-1); 
  19.                 if(!emptyempty($route)) { 
  20.                     $url    =   str_replace(APP_NAME,$app,).'/'.C('VAR_ROUTER').'/'.$route.'/'.$str; 
  21.                 }else{ 
  22.                     $url    =   str_replace(APP_NAME,$app,).'/'.C('VAR_MODULE').'/'.$module.'/'.C('VAR_ACTION').'/'.$action.$str; 
  23.                 } 
  24.                 break; 
  25.             case 2:// 智能PATHINFO模式 
  26.                 $depr   =   C('PATH_DEPR'); 
  27.                 $str    =   $depr; 
  28.                 foreach ($params as $var=>$val) 
  29.                     $str .= $var.$depr.$val.$depr; 
  30.                 $str = substr($str,0,-1); 
  31.                 if(!emptyempty($route)) { 
  32.                     $url    =   str_replace(APP_NAME,$app,).'/'.$route.$str; 
  33.                 }else{ 
  34.                     $url    =   str_replace(APP_NAME,$app,).'/'.$module.$depr.$action.$str; 
  35.                 } 
  36.                 break; 
  37.         } 
  38.         if(C('HTML_URL_SUFFIX')) { 
  39.             $url .= C('HTML_URL_SUFFIX'); 
  40.         } 
  41.     }else{ 
  42.         $params =   http_build_query($params); 
  43.         if(!emptyempty($route)) { 
  44.             $url    =   str_replace(APP_NAME,$app,).'?'.C('VAR_ROUTER').'='.$route.'&'.$params; 
  45.         }else{ 
  46.             $url    =   str_replace(APP_NAME,$app,).'?'.C('VAR_MODULE').'='.$module.'&'.C('VAR_ACTION').'='.$action.'&'.$params; 
  47.         } 
  48.     } 
  49.     return $url; 
  50. } 
  51. ?> 
PHP代码
   1. 
   2. require_once 'functions.php'; 
   3.  
   4. class functionsTest extends PHPUnit_Framework_TestCase 
   5. { 
   6.     /**
   7.      * 确认url()返回预期的字符串
   8.      */ 
   9.     public function testurl() 
  10.     { 
  11.         define('', 'index.php'); 
  12.  
  13.         C('VAR_MODULE', 'module'); 
  14.         C('VAR_ACTION', 'action'); 
  15.         C('VAR_ROUTER', 'route'); 
  16.  
  17.         // 测试通常模式URL 
  18.         $uri = url('Index', 'Home', '', APP_NAME, array('q' => 'test', 'msg' => 'OK')); 
  19.         $this->assertEquals('index.php?module=Home&action=Index&q=test&msg=OK', $uri); 
  20.  
  21.         // 测试通常模式路由 
  22.         $uri = url('Index', 'Home', 'default', APP_NAME, array('q' => 'test', 'msg' => 'OK')); 
  23.         $this->assertEquals('index.php?route=default&q=test&msg=OK', $uri); 
  24.  
  25.         
  26.         C('DISPATCH_ON', true); 
  27.         C('URL_MODEL', 1); 
  28.         C('PATH_MODEL', 1); 
  29.         
  30.         // 测试普通PATHINFO模式URL 
  31.         $uri = url('Index', 'Home', '', APP_NAME, array('q' => 'test', 'msg' => 'OK')); 
  32.         $this->assertEquals('index.php/module/Home/action/Index/q/test/msg/OK', $uri); 
  33.  
  34.         // 测试普通PATHINFO模式路由 
  35.         $uri = url('Index', 'Home', 'default', APP_NAME, array('q' => 'test', 'msg' => 'OK')); 
  36.         $this->assertEquals('index.php/route/default/q/test/msg/OK', $uri); 
  37.  
  38.         C('PATH_MODEL', 2); 
  39.         C('PATH_DEPR', '/'); 
  40.  
  41.         // 测试智能PATHINFO模式URL 
  42.         $uri = url('Index', 'Home', '', APP_NAME, array('q' => 'test', 'msg' => 'OK')); 
  43.         $this->assertEquals('index.php/Home/Index/q/test/msg/OK', $uri); 
  44.  
  45.         // 测试智能PATHINFO模式路由 
  46.         $uri = url('Index', 'Home', 'default', APP_NAME, array('q' => 'test', 'msg' => 'OK')); 
  47.         $this->assertEquals('index.php/default/q/test/msg/OK', $uri); 
  48.     } 
  49. } 
  50. ?> 
异常测试
有时程序执行了非法操作而抛出异常,我们需要模拟某个异常,然后捕捉它是否触发了该异常。
PHP代码
   1. 
   2. class UnitTest extends PHPUnit_Framework_TestCase 
   3. { 
   4.     /**
   5.      * 测试异常
   6.      */ 
   7.     public function testException() 
   8.     { 
   9.         // 期望Exception异常 
  10.         $this->setExpectedException('Exception'); 
  11.  
  12.         // 抛出Exception异常 
  13.         throw new Exception('TestException'); 
  14.     } 
  15. } 
  16. ?> 
输出测试
有时某个方法并不返回而输出某些内容,我们需要继承PHPUnit_Extensions_OutputTestCase类来捕捉输出内容。PHPUnit默认不载入扩展类,需要自己加载。
PHP代码
   1. 
   2. // 载入输出测试用例扩展 
   3. require_once 'PHPUnit/Extensions/OutputTestCase.php'; 
   4.  
   5. class UnitTest extends PHPUnit_Extensions_OutputTestCase 
   6. { 
   7.     /**
   8.      * 测试输出
   9.      */ 
  10.     public function testOutput() 
  11.     { 
  12.         // 期望输出的内容是字符串 'Hello' 
  13.         $this->expectOutputString('Hello'); 
  14.  
  15.         // 输出 'Hello' 
  16.         echo 'Hello'; 
  17.     } 
  18. } 
  19. ?> 
数据库测试
PHPUnit 的数据库测试并不完善,只提供了assertTablesEqual和assertDataSetsEqual两个断言与 createFlatXMLDataSet和createXMLDataSet创建XML数据集的方法。无法进行全面的数据操作测试,建议使用 DBUnit。
附录
PHPUnit断言参考
assertArrayHasKey($key, array $array, $message = '')
assertArrayNotHasKey($key, array $array, $message = '')
assertAttributeContains($needle, $haystackAttributeName, $haystackClassOrObject, $message = '')
assertAttributeContainsOnly($type, $haystackAttributeName, $haystackClassOrObject, $isNativeType = NULL, $message = '')
assertAttributeEquals($expected, $actualAttributeName, $actualClassOrObject, $message = '', $delta = 0, $maxDepth = 10, $canonicalizeEol = FALSE)
assertAttributeGreaterThan($expected, $actualAttributeName, $actualClassOrObject, $message = '')
assertAttributeGreaterThanOrEqual($expected, $actualAttributeName, $actualClassOrObject, $message = '')
assertAttributeLessThan($expected, $actualAttributeName, $actualClassOrObject, $message = '')
assertAttributeLessThanOrEqual($expected, $actualAttributeName, $actualClassOrObject, $message = '')
assertAttributeNotContains($needle, $haystackAttributeName, $haystackClassOrObject, $message = '')
assertAttributeNotContainsOnly($type, $haystackAttributeName, $haystackClassOrObject, $isNativeType = NULL, $message = '')
assertAttributeNotEquals($expected, $actualAttributeName, $actualClassOrObject, $message = '', $delta = 0, $maxDepth = 10, $canonicalizeEol = FALSE)
assertAttributeNotSame($expected, $actualAttributeName, $actualClassOrObject, $message = '')
assertAttributeSame($expected, $actualAttributeName, $actualClassOrObject, $message = '')
assertClassHasAttribute($attributeName, $className, $message = '')
assertClassHasStaticAttribute($attributeName, $className, $message = '')
assertClassNotHasAttribute($attributeName, $className, $message = '')
assertClassNotHasStaticAttribute($attributeName, $className, $message = '')
assertContains($needle, $haystack, $message = '')
assertContainsOnly($type, $haystack, $isNativeType = NULL, $message = '')
assertEqualXMLStructure(DOMNode $expectedNode, DOMNode $actualNode, $checkAttributes = FALSE, $message = '')
assertEquals($expected, $actual, $message = '', $delta = 0, $maxDepth = 10, $canonicalizeEol = FALSE)
assertFalse($condition, $message = '')
assertFileEquals($expected, $actual, $message = '', $canonicalizeEol = FALSE)
PHPUnit官方文档
http://www.phpunit.de/manual/3.6/en/index.html