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

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

      推薦:解析PHP編碼規范之注釋和文件結構
      文件結構 | |――images |――include |――parameter |――config |――function |――index images存放圖片文件,include中是系統是要引用的文件,一般在parameter中存放參數文件,config中存放配置文件,function中存放方法文件,如javascript的方法等,

      現在我們可以開始添加正確訪問數據庫的代碼 —— 一個方法一個方法地添加 —— 直到所有這 3 個測試都可以通過。最終版本的 dblib.php 代碼如下所示。

      清單 9. 完整的 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()

      {

      $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 頁面。

      這個頁面對兩個數字進行求和。為了對這個頁面進行測試,我們首先從單元測試代碼開始入手。

      清單 10. TestPage.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( '//', $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( '//', $page ) == 1 );

      preg_match( '/(.*?)<\/span>/', $page, $out );

      $this->assertTrue( $out[1]=='30' );

      }

      }

      ?>

      這個測試使用了 PEAR 提供的 HTTP Client 模塊。我發現它比內嵌的 PHP Client URL Library(CURL)更簡單一點兒,不過也可以使用后者。

      有一個測試會檢查所返回的頁面,并判斷這個頁面是否包含 HTML。第二個測試會通過將值放到請求的 URL 中來請求計算 10 和 20 的和,然后檢查返回的頁面中的結果。

      這個頁面的代碼如下所示。

      清單 11. TestPage.php

      =


       

      這個頁面相當簡單。兩個輸入域顯示了請求中提供的當前值。結果 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 點。

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

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