測試實踐:Eclipse 之 JUnit

測試實踐:Eclipse 之 JUnit(Using JUnit With Eclipse IDE)


這篇文章將給你介紹JUnit——一個工程測試調式的工具。 在介紹了了 測試驅動開發理論之後,我們繼續介紹“怎樣用Eclipse創建你的、JUnit Test”。 我們會用象"hello word"一樣簡單例子來向你揭露JUnit Case。

自動化測試(automated testing)在好多書籍中被介紹了,但很少注意講怎樣去組織這些測試。 當測試寫的越多時,很難知道把這些測試放到哪或者用什麼去調用它們。 在極限編程---Extreme Programming(xp),測試驅動開發 Test-Driven Development (TDD)盛行的時代,這成了一個很大的問題。 你可以把 測試驅動開發(TDD)認爲是"Development through testing" 開發由經測試。

TDD的主要條款:

  • 在任何代碼片段之前,必須先寫好自動檢測這段代碼功能的程序。既然代碼不存在,那麼測試在一開始就失敗。

  • 在 測試通過之後,複製的代碼必須刪掉。

象這樣的方式每個程序員都可以應用,並不需要特定的方法論。但在我們開始寫test之前, 值得我們注意的是,先考慮一下如何組織自動化測試。

這裏有幾種我們需要考慮的測試

  • 單元測試(Unit test) :這些是爲檢查個別模塊(比如classes類)服務的。 如果對象需要訪問外部的數據源,比如Database,就需要通過一些模擬的對象(MOCK object)來模擬Database, (但這也只有在真實環境的數據與測試環境不同的時候。
    比如測試環境裏面沒有真實Datebase,就需要MOCK Object)

  • 用戶測試 (Customer's test):這裏是功能的,系統的並且認可的測試。系統中所有的行爲檢查都做爲一個整體。 在XP理論中,這些測試,是由用戶編寫的,給出測試案例提綱。

  • 集成測試 (Itegration tests): 這些測試象是在用戶測試和單元測試之間的十字路口。 集成測試幫助程序測試幾個級別中交互。 ,Mock Object不會出現在集承測試中,他會增加測試時間。同樣,集成測試也經常需要存在的特定的測試環境,比如從數據庫中放一些測試數據。集成測試也許使用外部的lib。 Cactus就是這樣一個J2EE集成的lib。 解釋這些測試已經超出了本篇文章的範圍,並且也需要詳細的理論敘述,所以,你僅需要知道這種測試存在就可以了。

  • 開發測試(Developer's test) : 這種測試就是那些開發者校驗 整段代碼,新加的代碼,新加的函數函數。 對於每個開發而言, 隨時生成新的的測試去檢查代碼是很重要的。 組織這些測試和組織這些代碼有着同樣的重要性。

至於本文其他地方,只要說到"測試",就是專指開發測試(Developer's test)。

在開發期間, 一個程序員有時可能問自己:系統中這個行爲有test麼,這個test存在麼,哪裏可以找到這個test?每次發現錯誤,都是靠最基礎修改bug而不是通過自動測試,這是一個典型的例子。 在這種情形下事情進展可能是:

  1. 去找到這個函數的測試(可能測試已經寫了,但裏面還有一些小錯誤)
  2. 如果這樣的測試還沒有,或者測試不能蓋住這種錯誤,我們就寫一個新的測試來蓋住這種錯誤。
  3. 現在 我們深信,程序在新的測試中不會通過。
  4. 修復程序中的bug。
  5. 再運行測試
  6. 確定程序在測試中通過了。

當然,可能出現各種各樣的處理, 但思想必須很明確:你只需糾正那些被測試找出那些錯誤。

現在,讓我們告訴你一個開發人員怎樣解決這種情形。 通過存在的功能性的測試

  • 我利用一些集成的開發環境(IDE)來查找 被修正那些類和方法的放在什麼地方。

  • 製造一個已知的錯誤環境,來查找那些代碼判斷存在錯誤。

  • 最後但不是最不重要的,寫好測試並且放到一個現有的測試類中去。 如果你不小心出了錯誤, 期望你和你的同事能注意到副本,並且糾正它

都準就緒,開始建立測試了, 所以現在需要給測試取一個名稱。 你可能說,“這不是問題: 在每個類面前加個Test就是了!” 但並不是那麼簡單的, 讓我告訴你這樣如果可能造成的問題:

  • 當時候我們在使用TDD的方式開發時, 需要測試的class或者method可能都不存在。

  • 也可能一個test 含蓋了好幾個方法,甚至好幾個classes。

這些僅僅是最普通的問題, 下面還有更多。

給個在test命名上的建議: test 類的取名首先應該表達出這個類是一個test類,並且能確切的表示出他要檢查哪些,留有這個原class名的味道。 其實這很容易,請別擔心這個名稱會變的很長或者很醜陋,自己隨便怎樣取都可以。

下面我們將使用Eclipse中的JUnit工具建立我們的第一個測試,假定你已經下載了這個產品的當前版本, 如果沒有,你隨時可以從它的官方網站(www.eclipse.org)下載。我們需要JUnit,你也可以從它的官方網站(www.junit.org)上下載,下載並解壓縮到你硬盤中存放java libaries的地方。

打開Eclipse.我們將建立一個新的工程的工作空間(workplace project) 點 File -> New ->Project,選擇Java一路Next。 輸入工程名稱(project name),比如ProjectWithJUnit. 點擊完成。 這樣就建立了一個新工程,讓我們配置一下我們的Eclipse,於是,我們把JUnit library 添加到build path. 點擊 Project-->Properties, 選擇Java Build Path Libraries, 點Add Exteranal JARs 選中JUnit.jar。 你將會看到JUnit將會出現在的屏幕上 libraries列表中。 點Okay,Eclipse將強制rebuild所有的build paths.

我們已經準備好,開始寫我們的"Hello World"了 . 讓我們遵照TDD規範:在編碼之前就建立測試。爲了, 我們將假頂我們將要寫的類名是HelloWorld 有一個返回字符串的方法 say().

要建立這樣一個test, 在ProjectWithJUnit標題上右鍵, 選擇New -> Other,展開"Java", 選擇JUnit. 在對話框的右邊一攔裏選擇TestCase,接着點Next. 參見圖1。

圖1。 在Eclipse 中建立JUnit test

CSDN_Dev_Image_2004-2-101312130.gif

在Test class:一攔裏輸入我們需要測試的class--HelloWorld。並且給Test case取個名稱--- 比如,TestThatWeGetHelloWorldPrompt(是的,這看上去太長了,但是它能很清楚表達出它的意思) 點Finish完成。

下面是 TestThatWeGetHelloWorldPrompt.java的代碼:

public class TestThatWeGetHelloWorldPrompt

    extends TestCase {

    public TestThatWeGetHelloWorldPrompt(

        String name) {

        super(name);

    }

    public void testSay() {

        HelloWorld hi = new HelloWorld();

        assertEquals("Hello World!", hi.say());

    }

    public static void main(String[] args) {

        junit.textui.TestRunner.run(

            TestThatWeGetHelloWorldPrompt.class);

    }

}

這個代碼一點都不復雜,僅僅有一點點特別。 不管怎樣,讓我們詳細的檢查它。 我們繼承了JUnit的TestCase. (TestCase 在JUnit的javadoc裏定義是"用來運行多個Test的固定裝置")。 JUnit也定義了TestSuite 由於一組關聯的TestCase組成..


通過以下兩步來建立我們簡單的Test Case;

  1. 建立Junit.framework.TestCase的實例.
  2. 定義一些 以"test"開頭的測試函數, 並且返回一空值.(比如 testWasTranscationSuccessful(),testShow()等等).

TestThatWeGetHelloWorldPrompt.java 同時遵循這些標準: 這些TestCase的子類含有一個testSay()的方法. 這個方法由assertEquals()方法調用, 用於檢驗say()的返回值(按照這裏的做法返回應該是不一致,因爲一開始建立的HelloWorld 我們讓say()返回的值是null).

main()主函數是用來運行test並且顯示輸出的結果. JUnit的TestRunnery以(swing.u)圖形和本文(text.ui)的的方式來執行我們的test並反饋信息。我們就使用文本(text.ui),這個Eclipse肯定支持. (譯註:這裏可能翻譯的不怎麼好,所謂文本和圖形,是指你在建立TestCase的時候,有一個選項,Which method stubs would you like to create,選擇text.ui|| swing.ui||awt.ui,一般是選擇text.ui因爲Eclipse肯定支持這個), 依照這些文本的信息,Eclipse同時會生成圖形顯示。(在Package Exploer的下面Tab條上會多個JUnit,點它就看到了:)。

又一個所以,按照現在這樣測試驅動的開發的做法, 一旦我們跑起了我們的test,我們應該看到返回一些錯誤的信息。 點Run-> Run as -> JUnit Test(注意啊, 這個TestThatWeGetHelloWorldPrompt.java應該在Package Explorer被點中,在左邊那個window中),你點到的因該是JUnit window(就是下面的那個Tab條,注意不是Package Exploer),這樣你就看到了JUnit window, 他會顯示一個紅色條,表示是一個失敗的Test。 (如果你按了運行它沒有自動轉到這個窗口,你可以點做下Tab條 上的JUnit標籤。)

CSDN_Dev_Image_2004-2-11630400.gif

一按運行, 太好了,果然出錯了。好,現在正式開始建立用於工作的HelloWorld代碼,---點New->Class,可能和原來的的重複,那就把原來的刪掉。代碼如下:

HelloWolrd.java

public class HelloWorld {

    public String say() {

        return("Hello World!");

    }

}

這及爲簡單的,都用不着註釋。現在再來測試一下看看結果。就用上面的方法,點Run-> Run As Jnit. 在左邊的JUnit窗口中出現了一個綠條。 看圖三。 出現綠色的條表示測試通過了。

CSDN_Dev_Image_2004-2-11630402.gif

現在,再變個條件,讓測試不通過。 這將幫助我們理解JUnit test怎樣覆蓋並且報出不同的錯誤。 編輯 assertEquals()方法,把它的返回值從"Hello World!"變成另外一個值 比如"Hello ME!". 這樣,當你再運行這個JUnit test,那個顯示條又變成紅的了,並且在Failuer Trace裏看到是不是什麼導致了錯誤。 如圖:

CSDN_Dev_Image_2004-2-11630404.gif

總結。我想說一些自己的想法(這裏還是原文不是翻譯過來的)。 我過去並不認爲測試代碼是開發過程中很重要的一部分。 但在最近幾年發展的很快,多虧了那些方法論(比如基於異常開發"exceptions-based development"等),他們促進了測試以及測試工具的發展。

如果你對本文感興趣, 您可以花點時間正式的學習一下測試理論,把它應用到你的工作中去。.


學習測試一些資源:

JUint主頁

支持Struts Framework 的JUnit 組件:Newhttp://strutstestcase.sourceforge.net/

關於XP和網站Newhttp://www.chianxp.org

Martin Fowler《Refactoring》 中文《重構》侯傑譯。

總之,我覺得這是一篇 step by step 非常簡單的入門文章,很容易讓人明白,所以就厚着臉皮翻譯過來了。

Totodo譯,水平有限,有錯誤請及時指出,非常感謝.

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章