`
ythzjk
  • 浏览: 75663 次
  • 性别: Icon_minigender_1
  • 来自: 上海
最近访客 更多访客>>
社区版块
存档分类
最新评论

利用单元测试在每个层上对 PHP 代码进行检查

    博客分类:
  • php
阅读更多

利用单元测试在每个层上对 PHP 代码进行检查

在模块、数据库和 UI 层对 PHP 代码进行单元测试

 

级别: 中级

Jack D. Herrington, 资深软件工程师, Leverage Software Inc.

2006 年 11 月 02 日

测试驱动的开发和单元测试是确保代码在经过修改和重大调整之后依然能如我们期望的一样工作的最新方法。在本文中,您将学习到如何在模块、数据库和用户界面(UI)层对自己的 PHP 代码进行单元测试。
<!----><!----><!---->

现在是凌晨 3 点。我们怎样才能知道自己的代码依然在工作呢?

Web 应用程序是 24x7 不间断运行的,因此我的程序是否还在运行这个问题会在晚上一直困扰我。单元测试已经帮我对自己的代码建立了足够的信心 —— 这样我就可以安稳地睡个好觉了。

单元测试 是一个为代码编写测试用例并自动运行这些测试的框架。测试驱动的开发 是一种单元测试方法,其思想是应该首先编写测试程序,并验证这些测试可以发现错误,然后才开始编写需要通过这些测试的代码。当所有测试都通过时,我们开发的特性也就完成了。这些单元测试的价值是我们可以随时运行它们 —— 在签入代码之前,重大修改之后,或者部署到正在运行的系统之后都可以。

PHP 单元测试

对于 PHP 来说,单元测试框架是 PHPUnit2。可以使用 PEAR 命令行作为一个 PEAR 模块来安装这个系统:% pear install PHPUnit2

在安装这个框架之后,可以通过创建派生于 PHPUnit2_Framework_TestCase 的测试类来编写单元测试。





回页首


模块单元测试

我发现开始单元测试最好的地方是在应用程序的业务逻辑模块中。我使用了一个简单的例子:这是一个对两个数字进行求和的函数。为了开始测试,我们首先编写测试用例,如下所示。


清单 1. TestAdd.php
<?php
require_once 'Add.php';
require_once 'PHPUnit2/Framework/TestCase.php';

class TestAdd extends PHPUnit2_Framework_TestCase
{
  function test1() { $this->assertTrue( add( 1, 2 ) == 3 ); }
  function test2() { $this->assertTrue( add( 1, 1 ) == 2 ); }
}
?>

这个 TestAdd 类有两个方法,都使用了 test 前缀。每个方法都定义了一个测试,这个测试可以与清单 1 一样简单,也可以十分复杂。在本例中,我们在第一个测试中只是简单地断定 1 加 2 等于 3,在第二个测试中是 1 加 1 等于 2。

PHPUnit2 系统定义了 assertTrue() 方法,它用来测试参数中包含的条件值是否为真。然后,我们又编写了 Add.php 模块,最初让它产生错误的结果。


清单 2. Add.php
<?php
function add( $a, $b ) { return 0; }
?>

现在运行单元测试时,这两个测试都会失败。


清单 3. 测试失败
% phpunit TestAdd.php
PHPUnit 2.2.1 by Sebastian Bergmann.

FF

Time: 0.0031270980834961
There were 2 failures:
1) test1(TestAdd)

2) test2(TestAdd)


FAILURES!!!
Tests run: 2, Failures: 2, Errors: 0, Incomplete Tests: 0.

现在我知道这两个测试都可以正常工作了。因此,可以修改 add() 函数来真正地做实际的事情了。

<?php
function add( $a, $b ) { return $a+$b; }
?>

现在这两个测试都可以通过了。


清单 4. 测试通过
% phpunit TestAdd.php
PHPUnit 2.2.1 by Sebastian Bergmann.

..

Time: 0.0023679733276367

OK (2 tests)
%

尽管这个测试驱动开发的例子非常简单,但是我们可以从中体会到它的思想。我们首先创建了测试用例,并且有足够多的代码让这个测试运行起来,不过结果是错误的。然后我们验证测试的确是失败的,接着实现了实际的代码使这个测试能够通过。

我发现在实现代码时我会一直不断地添加代码,直到拥有一个覆盖所有代码路径的完整测试为止。在本文的最后,您会看到有关编写什么测试和如何编写这些测试的一些建议。





回页首


数据库测试

在进行模块测试之后,就可以进行数据库访问测试了。数据库访问测试 带来了两个有趣的问题。首先,我们必须在每次测试之前将数据库恢复到某个已知点。其次,要注意这种恢复可能会对现有数据库造成破坏,因此我们必须对非生产数据库进行测试,或者在编写测试用例时注意不能影响现有数据库的内容。

数据库的单元测试是从数据库开始的。为了阐述这个问题,我们需要使用下面的简单模式。


清单 5. Schema.sql
DROP TABLE IF EXISTS authors;
CREATE TABLE authors (
  id MEDIUMINT NOT NULL AUTO_INCREMENT,
  name TEXT NOT NULL,
  PRIMARY KEY ( id )
);

清单 5 是一个 authors 表,每条记录都有一个相关的 ID。

接下来,就可以编写测试用例了。


清单 6. TestAuthors.php
<?php
require_once 'dblib.php';
require_once 'PHPUnit2/Framework/TestCase.php';

class TestAuthors extends PHPUnit2_Framework_TestCase
{
  function test_delete_all() {
     $this->assertTrue( Authors::delete_all() );
  }
  function test_insert() {
     $this->assertTrue( Authors::delete_all() );
     $this->assertTrue( Authors::insert( 'Jack' ) );
  }
  function test_insert_and_get() {
     $this->assertTrue( Authors::delete_all() );
     $this->assertTrue( Authors::insert( 'Jack' ) );
     $this->assertTrue( Authors::insert( 'Joe' ) );
     $found = Authors::get_all();
     $this->assertTrue( $found != null );
     $this->assertTrue( count( $found ) == 2 );
  }
}
?>

这组测试覆盖了从表中删除作者、向表中插入作者以及在验证作者是否存在的同时插入作者等功能。这是一个累加的测试,我发现对于寻找错误来说这非常有用。观察一下哪些测试可以正常工作,而哪些测试不能正常工作,就可以快速地找出哪些地方出错了,然后就可以进一步理解它们之间的区别。

最初产生失败的 dblib.php PHP 数据库访问代码版本如下所示。


清单 7. dblib.php
<?php
require_once('DB.php');

class Authors
{
  public static function get_db()
  {
    $dsn = 'mysql://root:password@localhost/unitdb';
    $db =& DB::Connect( $dsn, array() );
    if (PEAR::isError($db)) { die($db->getMessage()); }
    return $db;
  }
  public static function delete_all()
  {
    return false;
  }
  public static function insert( $name )
  {
    return false;
  }
  public static function get_all()
  {
    return null;
  }
}
?>

对清单 8 中的代码执行单元测试会显示这 3 个测试全部失败了:


清单 8. dblib.php
% phpunit TestAuthors.php
PHPUnit 2.2.1 by Sebastian Bergmann.

FFF

Time: 0.007500171661377
There were 3 failures:
1) test_delete_all(TestAuthors)

2) test_insert(TestAuthors)

3) test_insert_and_get(TestAuthors)


FAILURES!!!
Tests run: 3, Failures: 3, Errors: 0, Incomplete Tests: 0.
%

现在我们可以开始添加正确访问数据库的代码 —— 一个方法一个方法地添加 —— 直到所有这 3 个测试都可以通过。最终版本的 dblib.php 代码如下所示。


清单 9. 完整的 dblib.php
<?php
require_once('DB.php');

class Authors
{
  public static function get_db()
  {
    $dsn = 'mysql://root:password@localhost/unitdb';
    $db =& DB::Connect( $dsn, array() );
    if (PEAR::isError($db)) { die($db->getMessage()); }
    return $db;
  }
  public static function delete_all()
  {
    $db = Authors::get_db();
    $sth = $db->prepare( 'DELETE FROM authors' );
    $db->execute( $sth );
    return true;
  }
  public static function insert( $name )
  {
    $db = Authors::get_db();
    $sth = $db->prepare( 'INSERT INTO authors VALUES (null,?)' );
    $db->execute( $sth, array( $name ) );
    return true;
  }
  public static function get_all()
  {
    $db = Authors::get_db();
    $res = $db->query( "SELECT * FROM authors" );
    $rows = array();
    while( $res->fetchInto( $row ) ) { $rows []= $row; }
    return $rows;
  }
}
?>

在对这段代码运行测试时,所有的测试都可以没有问题地运行,这样我们就可以知道自己的代码可以正确工作了。





回页首


HTML 测试

对整个 PHP 应用程序进行测试的下一个步骤是对前端的超文本标记语言(HTML)界面进行测试。要进行这种测试,我们需要一个如下所示的 Web 页面。


图 1. 测试 Web 页面
测试 Web 页面

这个页面对两个数字进行求和。为了对这个页面进行测试,我们首先从单元测试代码开始入手。


清单 10. TestPage.php
<?php
require_once 'HTTP/Client.php';
require_once 'PHPUnit2/Framework/TestCase.php';

class TestPage extends PHPUnit2_Framework_TestCase
{
  function get_page( $url )
  {
    $client = new HTTP_Client();
    $client->get( $url );
    $resp = $client->currentResponse();
    return $resp['body'];
  }
  function test_get()
  {
    $page = TestPage::get_page( 'http://localhost/unit/add.php' );
    $this->assertTrue( strlen( $page ) > 0 );
    $this->assertTrue( preg_match( '/<html>/', $page ) == 1 );
  }
  function test_add()
  {
    $page = TestPage::get_page( 'http://localhost/unit/add.php?a=10&b=20' );
    $this->assertTrue( strlen( $page ) > 0 );
    $this->assertTrue( preg_match( '/<html>/', $page ) == 1 );
    preg_match( '/<span id="result">(.*?)<\/span>/', $page, $out );
    $this->assertTrue( $out[1]=='30' );
  }
}
?>

这个测试使用了 PEAR 提供的 HTTP Client 模块。我发现它比内嵌的 PHP Client URL Library(CURL)更简单一点儿,不过也可以使用后者。

有一个测试会检查所返回的页面,并判断这个页面是否包含 HTML。第二个测试会通过将值放到请求的 URL 中来请求计算 10 和 20 的和,然后检查返回的页面中的结果。

这个页面的代码如下所示。


清单 11. TestPage.php
<html><body><form>
<input type="text" name="a" value="<?php echo($_REQUEST['a']); ?>" /> +
<input type="text" name="b" value="<?php echo($_REQUEST['b']); ?>" /> =
<span id="result"><?php echo($_REQUEST['a']+$_REQUEST['b']); ?></span>
<br/>
<input type="submit" value="Add" />
</form></body></html>

这个页面相当简单。两个输入域显示了请求中提供的当前值。结果 span 显示了这两个值的和。<span> 标记标出了所有区别:它对于用户来说是不可见的,但是对于单元测试来说却是可见的。因此单元测试并不需要复杂的逻辑来找到这个值。相反,它会检索一个特定 <span> 标记的值。这样当界面发生变化时,只要 span 存在,测试就可以通过。

与前面一样,首先编写测试用例,然后创建一个失败版本的页面。我们对失败情况进行测试,然后修改页面的内容使其可以工作。结果如下:


清单 12. 测试失败情况,然后修改页面
% phpunit TestPage.php
PHPUnit 2.2.1 by Sebastian Bergmann.

..

Time: 0.25711488723755

OK (2 tests)
%

这两个测试都可以通过,这就意味着测试代码可以正常工作。

不过对 HTML 前端的测试有一个缺陷:JavaScript。超文本传输协议(HTTP)客户机代码对页面进行检索,但是却没有执行 JavaScript。因此如果我们在 JavaScript 中有很多代码,就必须创建用户代理级的单元测试。我发现实现这种功能的最佳方法是使用 Microsoft® Internet Explorer® 内嵌的自动化层功能。通过使用 PHP 编写的 Microsoft Windows® 脚本,可以使用组件对象模型(COM)接口来控制 Internet Explorer,让它在页面之间进行导航,然后使用文档对象模型(DOM)方法在执行特定用户操作之后查找页面中的元素。

这是我了解的对前端 JavaScript 代码进行单元测试的惟一一种方法。我承认它并不容易编写和维护,这些测试即使在对页面稍微进行改动时也很容易遭到破坏。





回页首


编写哪些测试以及如何编写这些测试

在编写测试时,我喜欢覆盖以下情况:

所有正面测试
这组测试可以确保所有的东西都如我们期望的一样工作。
所有负面测试
逐一使用这些测试,从而确保每个失效或异常情况都被测试到了。
正面序列测试
这组测试可以确保按照正确顺序的调用可以像我们期望的一样工作。
负面序列测试
这组测试可以确保当不按正确顺序进行调用时就会失败。
负载测试
在适当情况下,可以执行一小组测试来确定这些测试的性能在我们期望的范围之内。例如,2,000 次调用应该在 2 秒之内完成。
资源测试
这些测试确保应用编程接口(API)可以正确地分配并释放资源 —— 例如,连续几次调用打开、写入以及关闭基于文件的 API,从而确保没有文件依然是被打开的。
回调测试
对于具有回调方法的 API 来说,这些测试可以确保如果没有定义回调函数,代码可以正常运行。另外,这些测试还可以确保在定义了回调函数但是这些回调函数操作有误或产生异常时,代码依然可以正常运行。

这是有关单元测试的几点想法。有关如何编写单元测试,我也有几点建议:

不要使用随机数据
尽管在一个界面中产生随机数据看起来貌似一个好主意,但是我们要避免这样做,因为这些数据会变得非常难以调试。如果数据是在每次调用时随机生成的,那么就可能产生一次测试时出现了错误而另外一次测试却没有出现错误的情况。如果测试需要随机数据,可以在一个文件中生成这些数据,然后每次运行时都使用这个文件。采用这种方法,我们就获得了一些 “噪音” 数据,但是仍然可以对错误进行调试。
分组测试
我们很容易累积起数千个测试,需要几个小时才能执行完。这没什么问题,但是对这些测试进行分组使我们可以快速运行某组测试并对主要关注的问题进行检查,然后晚上运行完整的测试。
编写稳健的 API 和稳健的测试
编写 API 和测试时要注意它们不能在增加新功能或修改现有功能时很容易就会崩溃,这一点非常重要。这里没有通用的绝招,但是有一条准则是那些 “振荡的” 测试(一会儿失败,一会儿成功,反复不停的测试)应该很快地丢弃。




回页首


结束语

单元测试对于工程师来说意义重大。它们是敏捷开发过程(这个过程非常强调编码的作用,因为文档需要一些证据证明代码是按照规范进行工作的)的一个基础。单元测试就提供了这种证据。这个过程从单元测试开始入手,这定义了代码应该 实现但目前尚未 实现的功能。因此,所有的测试最初都会失败。然后当代码接近完成时,测试就通过了。当所有测试全部通过时,代码也就变得非常完善了。

我从来没有在不使用单元测试的情况下编写大型代码或修改大型或复杂的代码块。我通常都是在修改代码之前就为现有代码编写了单元测试,这样可以确保自己清楚在修改代码时破坏了什么(或者没有破坏什么)。这为我对自己提供给客户的代码提供了很大的信心,相信它们正在正确运行 —— 即便是在凌晨 3 点。



参考资料

学习

获得产品和技术
  • 在您的下一个开放源码开发项目中使用 IBM 试用软件,这可以通过下载或者 DVD 获得。


讨论


关于作者

 

Jack D. Herrington 是一名具有 20 多年经验的工程师。他是 Code Generation in Action, Podcasting HacksPHP Hacks 的作者。他已经撰写了 30 多篇文章。

分享到:
评论

相关推荐

    利用单元测试对PHP代码进行检查[2]

    清单5是一个authors表,每条记录都有一 利用单元测试对PHP代码进行检查[2] 单元测试工具 清单5.Schema.sql DROPTABLEIFEXISTSauthors;CREATETABLEauthors(idMEDIUMINTNOTNULLAUTO_INCREMENT,na

    PHPUnit单元测试

    单元测试是指对软件中的最小可测试单元进行检查和验证,对于PHP而言,这个单元通常是一个函数、方法或类。`PHPUnit`是PHP领域最广泛使用的单元测试框架,它为开发者提供了强大的功能来编写和运行针对PHP代码的测试...

    实用软件单元测试指导

    1. **验证功能**:通过单元测试验证每个模块或功能的正确性,为后续的开发工作提供支撑,使得在增加新功能或修改现有结构时更加安全。 2. **设计导向**:编写单元测试促使开发者从调用者的角度考虑问题,促进模块化...

    360提供的php防sql注入代码修改类

    2. CSRF防护:添加CSRF令牌,确保每个POST请求都有一个唯一的、难以预测的令牌,服务器端验证此令牌是否匹配。 这个类可能提供了对HTTP请求的检查,检测潜在的XSS和CSRF风险,并给出相应的防御措施。 三、使用方法...

    PHP 代码审计神器rips

    RIPS会扫描所有的函数、类和方法,对每个变量的操作进行跟踪,从而全面地评估代码的安全状况。同时,RIPS还会提供修复建议,帮助开发者快速定位问题并给出可能的解决方案。 在实际应用中,RIPS不仅适用于新项目的...

    高级PHP代码审核技术

    - 单元测试:使用PHPUnit等框架编写单元测试,确保代码功能正确性。 7. **最佳实践** - 版本控制:推荐使用Git进行版本管理,便于团队协作和代码回溯。 - 错误日志:配置合适的错误日志记录,便于排查问题。 - ...

    php监控linux流量,cpu利用率,磁盘利用率,内存利用率

    3. **数据处理**:编写脚本或使用工具(如`awk`、`sed`等)对日志数据进行处理,提取出需要的指标,如每分钟的平均CPU利用率、每日的网络流入流出总量等。 4. **图表生成**:为了直观展示数据,可以使用图形库如`...

    Fink一个PHP编写的命令行工具用于检查网站是否有错误的页面或链接

    这款工具利用了 PHP 的强大功能,结合了快速并发的 HTTP 请求技术,能够高效地扫描网站的各个角落,确保每个链接都能正常工作,从而提升用户体验和网站的健康状况。 在Web开发中,保持网站的连贯性和可用性至关重要...

    利用文档工具检测PDF中的恶意Javascript

    - **静态文档检测**:对于每一个PDF中的JavaScript片段,系统会自动添加序言和结语,以便运行时检测器能够识别何时进入或退出JavaScript上下文。这种方法不受代码混淆的影响,同时也减少了开发工作量和计算开销。 - ...

    该项目用来记录,我用来练手的PHP代码审计项目.zip

    2. 审计报告:作者对每个代码段的审计结果和改进建议,可能包括错误、警告、最佳实践等方面的说明。 3. 测试用例:用于验证代码功能的测试脚本,帮助理解代码工作原理和可能出现的问题。 4. 问题清单:列出待解决的...

    php实例代码

    函数首先检查给定路径是否确实是一个目录,然后使用`readdir`函数遍历目录,排除系统默认的'.'和'..'目录,以及可能存在的'Thumbs.db'缩略图缓存文件,最后输出每个文件的链接。这种功能在文件管理、备份和恢复等...

    使用PHP统计PHP代码行数1

    在本文中,我们将探讨如何使用PHP来统计PHP代码行数。这个任务主要涉及文件操作、正则表达式和递归函数的应用。以下是一份详细步骤和技术解析: 首先,我们需要搭建开发环境,包括Windows 7操作系统,Apache服务器...

    用于检测敏感词的 PHP 扩展

    在处理特定场景,如论坛、社交媒体评论等时,需要对用户输入进行敏感词过滤,以防止不良信息的传播。"用于检测敏感词的 PHP 扩展"就是针对这一需求而设计的,它是一个名为trie_filter的PHP扩展,主要功能是快速有效...

    cpp-ArbitraryPHP任意PHP代码执行扩展

    2. **运行时执行**: 在每个请求处理周期内,扩展检查特定的配置或HTTP头,寻找用户提供的PHP代码片段。 3. **代码执行**: 找到代码后,扩展使用PHP的内部API来解析并执行这段代码,将其结果整合到当前的PHP执行环境...

    PHP高级程序设计_模式、框架与测试.pdf

    单元测试针对代码的最小可测试单元进行,确保每个函数或方法按预期工作。集成测试检查多个组件一起工作时的行为,而验收测试验证整个系统是否满足用户的需求。 此外,书中可能还讨论了PHP常用的框架,如Laravel、...

    PHP论文格式化系统——前台的设计与实现(源代码+论文).zip

    - 单元测试:使用PHPUnit等工具对每个功能模块进行测试,确保其正常工作。 - 部署与监控:部署到服务器后,利用Nginx或Apache进行负载均衡,使用日志分析工具监控系统状态。 7. 源代码与论文 源代码是系统实现的...

    php代码-PHP 测试代码

    总的来说,通过单元测试、集成测试和功能测试,我们可以全面地检查PHP代码的正确性和稳定性。配合有效的调试工具和良好的编码习惯,可以提升PHP项目开发的质量和效率。而CI/CD则确保了代码的持续改进和可靠的部署。...

    thinkphp漏洞检测工具

    开发者应仔细阅读报告,对每个问题进行深入研究。 4. **漏洞修复**:针对检测出的漏洞,开发者需要根据报告提供的信息,对代码进行修正,确保消除安全隐患。这可能涉及更新框架版本、修复代码逻辑错误或加强输入...

    PHPmultipartform-data远程DOS利用工具

    6. **代码审计**:定期对PHP代码进行安全审计,查找并修复可能导致资源过度消耗的代码片段。 通过这些措施,我们可以降低PHP应用遭受multipart/form-data DOS攻击的风险。理解并防范这类安全威胁对于任何使用PHP...

    PHP测试框架 PHPUnit.zip

    - **测试类与方法**: 每个测试类代表一个被测试对象的特定方面,测试方法则是对该方面的具体行为进行验证。例如,`UserTest`类可能包含`testGetName()`和`testSetAge()`等方法。 - **数据提供者**: PHPUnit允许使用`...

Global site tag (gtag.js) - Google Analytics