單元測試對PHP代碼的檢查_PHP教程

      編輯Tag賺U幣
      教程Tag:暫無Tag,歡迎添加,賺取U幣!

      推薦:解析PHP編碼規(guī)范之注釋和文件結(jié)構(gòu)
      文件結(jié)構(gòu) | |――images |――include |――parameter |――config |――function |――index images存放圖片文件,include中是系統(tǒng)是要引用的文件,一般在parameter中存放參數(shù)文件,config中存放配置文件,function中存放方法文件,如javascript的方法等,

      測試驅(qū)動的開發(fā)和單元測試是確保代碼在經(jīng)過修改和重大調(diào)整之后依然能如我們期望的一樣工作的最新方法。在本文中,您將學(xué)習(xí)到如何在模塊、數(shù)據(jù)庫和用戶界面(UI)層對自己的 PHP 代碼進(jìn)行單元測試。

      現(xiàn)在是凌晨 3 點(diǎn)。我們怎樣才能知道自己的代碼依然在工作呢?

      Web 應(yīng)用程序是 24x7 不間斷運(yùn)行的,因此我的程序是否還在運(yùn)行這個(gè)問題會在晚上一直困擾我。單元測試已經(jīng)幫我對自己的代碼建立了足夠的信心 —— 這樣我就可以安穩(wěn)地睡個(gè)好覺了。

      單元測試 是一個(gè)為代碼編寫測試用例并自動運(yùn)行這些測試的框架。測試驅(qū)動的開發(fā) 是一種單元測試方法,其思想是應(yīng)該首先編寫測試程序,并驗(yàn)證這些測試可以發(fā)現(xiàn)錯誤,然后才開始編寫需要通過這些測試的代碼。當(dāng)所有測試都通過時(shí),我們開發(fā)的特性也就完成了。這些單元測試的價(jià)值是我們可以隨時(shí)運(yùn)行它們 —— 在簽入代碼之前,重大修改之后,或者部署到正在運(yùn)行的系統(tǒng)之后都可以。

      PHP 單元測試

      對于 PHP 來說,單元測試框架是 PHPUnit2。可以使用 PEAR 命令行作為一個(gè) PEAR 模塊來安裝這個(gè)系統(tǒng):% pear install PHPUnit2。

      在安裝這個(gè)框架之后,可以通過創(chuàng)建派生于 PHPUnit2_Framework_TestCase 的測試類來編寫單元測試。

      模塊單元測試

      我發(fā)現(xiàn)開始單元測試最好的地方是在應(yīng)用程序的業(yè)務(wù)邏輯模塊中。我使用了一個(gè)簡單的例子:這是一個(gè)對兩個(gè)數(shù)字進(jìn)行求和的函數(shù)。為了開始測試,我們首先編寫測試用例,如下所示。

      清單 1. TestAdd.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 ); }

      }

      ?>

      這個(gè) TestAdd 類有兩個(gè)方法,都使用了 test 前綴。每個(gè)方法都定義了一個(gè)測試,這個(gè)測試可以與清單 1 一樣簡單,也可以十分復(fù)雜。在本例中,我們在第一個(gè)測試中只是簡單地?cái)喽?1 加 2 等于 3,在第二個(gè)測試中是 1 加 1 等于 2。

      PHPUnit2 系統(tǒng)定義了 assertTrue() 方法,它用來測試參數(shù)中包含的條件值是否為真。然后,我們又編寫了 Add.php 模塊,最初讓它產(chǎn)生錯誤的結(jié)果。

      清單 2. Add.php

       

      function add( $a, $b ) { return 0; }

      ?>

      現(xiàn)在運(yùn)行單元測試時(shí),這兩個(gè)測試都會失敗。

      清單 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)在我知道這兩個(gè)測試都可以正常工作了。因此,可以修改 add() 函數(shù)來真正地做實(shí)際的事情了。

       

      function add( $a, $b ) { return $a $b; }

      ?>

      現(xiàn)在這兩個(gè)測試都可以通過了。

      清單 4. 測試通過

      % phpunit TestAdd.php

      PHPUnit 2.2.1 by Sebastian Bergmann.

      ..

      Time: 0.0023679733276367

      OK (2 tests)

      %

      盡管這個(gè)測試驅(qū)動開發(fā)的例子非常簡單,但是我們可以從中體會到它的思想。我們首先創(chuàng)建了測試用例,并且有足夠多的代碼讓這個(gè)測試運(yùn)行起來,不過結(jié)果是錯誤的。然后我們驗(yàn)證測試的確是失敗的,接著實(shí)現(xiàn)了實(shí)際的代碼使這個(gè)測試能夠通過。

      我發(fā)現(xiàn)在實(shí)現(xiàn)代碼時(shí)我會一直不斷地添加代碼,直到擁有一個(gè)覆蓋所有代碼路徑的完整測試為止。在本文的最后,您會看到有關(guān)編寫什么測試和如何編寫這些測試的一些建議。

      數(shù)據(jù)庫測試

      在進(jìn)行模塊測試之后,就可以進(jìn)行數(shù)據(jù)庫訪問測試了。數(shù)據(jù)庫訪問測試 帶來了兩個(gè)有趣的問題。首先,我們必須在每次測試之前將數(shù)據(jù)庫恢復(fù)到某個(gè)已知點(diǎn)。其次,要注意這種恢復(fù)可能會對現(xiàn)有數(shù)據(jù)庫造成破壞,因此我們必須對非生產(chǎn)數(shù)據(jù)庫進(jìn)行測試,或者在編寫測試用例時(shí)注意不能影響現(xiàn)有數(shù)據(jù)庫的內(nèi)容。

      數(shù)據(jù)庫的單元測試是從數(shù)據(jù)庫開始的。為了闡述這個(gè)問題,我們需要使用下面的簡單模式。

      清單 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 是一個(gè) authors 表,每條記錄都有一個(gè)相關(guān)的 ID。

      接下來,就可以編寫測試用例了。

      清單 6. TestAuthors.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 );

      }

      }

      ?>

      這組測試覆蓋了從表中刪除作者、向表中插入作者以及在驗(yàn)證作者是否存在的同時(shí)插入作者等功能。這是一個(gè)累加的測試,我發(fā)現(xiàn)對于尋找錯誤來說這非常有用。觀察一下哪些測試可以正常工作,而哪些測試不能正常工作,就可以快速地找出哪些地方出錯了,然后就可以進(jìn)一步理解它們之間的區(qū)別。

      最初產(chǎn)生失敗的 dblib.php PHP 數(shù)據(jù)庫訪問代碼版本如下所示。

      清單 7. dblib.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 中的代碼執(zhí)行單元測試會顯示這 3 個(gè)測試全部失敗了:

      清單 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.

      %

      分享:解答PHP和MySQL操作應(yīng)該注意的一些細(xì)節(jié)
      對于 MySQL ,第一件你必須牢記的是它的每一行命令都是用分號 (;) 作為結(jié)束的,但沒有完全絕對的事,在這兒也是一樣。 當(dāng)一行 MySQL 被插入在 PHP 代碼中時(shí),最好把后面的分號省略掉,例如: mysql_query (INSERT INTO tablename (first_name, last_name) V

      共2頁上一頁12下一頁
      來源:模板無憂//所屬分類:PHP教程/更新時(shí)間:2009-06-21
      相關(guān)PHP教程