測試驅(qū)動的開發(fā)和單元測試是確保代碼在經(jīng)過修改和重大調(diào)整之后依然能如我們期望的一樣工作的新方法。在本文中,您將學習到如何在模塊、數(shù)據(jù)庫和用戶界面(UI)層對自己的 PHP 代碼進行單元測試。
現(xiàn)在是凌晨 3 點。我們怎樣才能知道自己的代碼依然在工作呢?
Web 應用程序是 24x7 不間斷運行的,因此我的程序是否還在運行這個問題會在晚上一直困擾我。單元測試已經(jīng)幫我對自己的代碼建立了足夠的信心 —— 這樣我可以安穩(wěn)地睡個好覺了。
單元測試 是一個為代碼編寫測試用例并自動運行這些測試的框架。測試驅(qū)動的開發(fā) 是一種單元測試方法,其思想是應該首先編寫測試程序,并驗證這些測試可以發(fā)現(xiàn)錯誤,然后才開始編寫需要通過這些測試的代碼。當所有測試都通過時,我們開發(fā)的特性也完成了。這些單元測試的價值是我們可以隨時運行它們 —— 在簽入代碼之前,重大修改之后,或者部署到正在運行的系統(tǒng)之后都可以。
PHP 單元測試
對于 PHP 來說,單元測試框架是 PHPUnit2?梢允褂 PEAR 命令行作為一個 PEAR 模塊來安裝這個系統(tǒng):% pear install PHPUnit2。
在安裝這個框架之后,可以通過創(chuàng)建派生于 PHPUnit2_Framework_TestCase 的測試類來編寫單元測試。
模塊單元測試
我發(fā)現(xiàn)開始單元測試好的地方是在應用程序的業(yè)務邏輯模塊中。我使用了一個簡單的例子:這是一個對兩個數(shù)字進行求和的函數(shù)。為了開始測試,我們首先編寫測試用例,如下所示。
清單 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 系統(tǒng)定義了 assertTrue() 方法,它用來測試參數(shù)中包含的條件值是否為真。然后,我們又編寫了 Add.php 模塊,初讓它產(chǎn)生錯誤的結果。
清單 2. Add.php
<?php
function add( $a, $b ) { return 0; }
?>
現(xiàn)在運行單元測試時,這兩個測試都會失敗。
清單 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.
現(xiàn)在我知道這兩個測試都可以正常工作了。因此,可以修改 add() 函數(shù)來真正地做實際的事情了。
<?php
function add( $a, $b ) { return $a+$b; }
?>
現(xiàn)在這兩個測試都可以通過了。
清單 4. 測試通過
% phpunit TestAdd.php
PHPUnit 2.2.1 by Sebastian Bergmann.
..
Time: 0.0023679733276367
OK (2 tests)
%
盡管這個測試驅(qū)動開發(fā)的例子非常簡單,但是我們可以從中體會到它的思想。我們首先創(chuàng)建了測試用例,并且有足夠多的代碼讓這個測試運行起來,不過結果是錯誤的。然后我們驗證測試的確是失敗的,接著實現(xiàn)了實際的代碼使這個測試能夠通過。
我發(fā)現(xiàn)在實現(xiàn)代碼時我會一直不斷地添加代碼,直到擁有一個覆蓋所有代碼路徑的完整測試為止。在本文的后,您會看到有關編寫什么測試和如何編寫這些測試的一些建議。