隨筆 - 3, 文章 - 152, 評(píng)論 - 17, 引用 - 0
          數(shù)據(jù)加載中……

          單元測試-入門篇

          前面的一篇文章(單元測試-理論篇)討論了什么是單元測試、單元測試的優(yōu)點(diǎn)并列舉了很多不寫單元測試的借口。如果你同意我們的觀點(diǎn),認(rèn)同單元測試確實(shí)是軟件開發(fā)中不可缺少的過程,那么我們就開始單元測試之旅吧!


          一個(gè)比較最大值的函數(shù)
            我們首先引入一個(gè)比較最大值的函數(shù)。我們傳入一個(gè)類型為int的數(shù)組參數(shù),它將返回最大值的那個(gè)元素。代碼如下:
          public class Largest {
           public static int largest(int[] datas){
            int max = 0;
            
            for(int i = 0 ; i < datas.length ; i++){
             if(max < datas[i]){
              max = datas[i];
             }
            }
            return max;
           }
          }
            可是,如何寫我們的測試代碼呢?


          直接在Largest類中添加一個(gè)main方法,要么重新寫一個(gè)可運(yùn)行的類來測試Largest。這樣的測試,同樣給我們帶來了很大的挑戰(zhàn):
          1、 驗(yàn)證困難。如何去驗(yàn)證代碼的行為和我們的期望一致呢?使用很多的if…else再加上==或equals()來判斷?對(duì)異常的情況又如何處理呢?混亂的驗(yàn)證,很容易給我們的測試代碼帶來BUG,讓我們對(duì)自己的測試不夠自信。
          2、 測試類無法管理。我們?nèi)绾沃庇^的得到測試運(yùn)行成功或失敗的消息?用原始的System.out.println()嗎?我們能一次運(yùn)行多個(gè)單元測試嗎?如果前面的測試運(yùn)行出現(xiàn)異常,后面的測試還能繼續(xù)運(yùn)行嗎?如果測試類很多,上百個(gè)甚至更多,我們能方便的由控制臺(tái)輸出測試結(jié)果嗎?
          3、 無法統(tǒng)計(jì)測試代碼覆蓋情況。缺少統(tǒng)一的測試代碼編寫規(guī)范和約定,可讀性和維護(hù)性差。
          不過,面對(duì)這些挑戰(zhàn)不用沮喪。單元測試框架已經(jīng)幫我們解決了這些問題,它提供了很多測試的基礎(chǔ)設(shè)施,讓我們能把更多的經(jīng)歷投入到測試代碼的編寫中來。


          JUnit
          JUnit最初是由Erich Gamma(GoF之一)和Kent Beck(xp和refactor的先驅(qū)之一)編寫的,它是一個(gè)開源的Java測試框架,用于編寫和運(yùn)行可重復(fù)的測試。


          下面我們逐步介紹如何對(duì)Largest類測試:
          一、 JUnit的安裝。如果你使用的開發(fā)工具是Eclipse,不用做任何安裝,它已經(jīng)提供了Junit的支持。否則,你需要去http://www.junit.org/ 下載Junit安裝包。安裝非常簡單,只要將junit.jar包設(shè)置到ClassPath中,讓你的Java代碼能夠找到它就可以了。
          二、 編寫測試代碼。代碼如下:
              public class LargestTest extends TestCase {
            public void testLargest(){
             int[] datas = {7,8,9};
             assertEquals(9,Largest.largest(datas));
            }
          }
          說明:
          1、 測試類一般要繼承抽象類TestCase。它實(shí)現(xiàn)了各種測試方法,并提供了一個(gè)測試過程的架構(gòu)。
          2、 測試代碼通過斷言(Assert)來判斷某個(gè)被測試函數(shù)是否正常工作。JUnit提供了很多斷言函數(shù),用來確定:某個(gè)條件是否為真;兩個(gè)數(shù)據(jù)是否相等,或者不等,或者其它的一些情況。
          3、 測試方法名以“test”開頭,這樣JUnit框架會(huì)自動(dòng)發(fā)現(xiàn)這是一個(gè)測試方法。

          三、 運(yùn)行測試類。
             運(yùn)行測試成功。

          我們的單元測試這樣就算完成了嗎?不,上面的測試只能算是一次驗(yàn)證而已。我們給的數(shù)據(jù)中,最大值9是數(shù)組的最后一個(gè)元素,如果9是第一個(gè)元素它還正確嗎?如果數(shù)據(jù)是負(fù)數(shù)呢?等等。我們的求最大值函數(shù)有著很多的邊界情況需要單元測試來驗(yàn)證。
             因此,我們?cè)趯憜卧獪y試之前,一定要對(duì)測試做一個(gè)周全的計(jì)劃,預(yù)先設(shè)置好要測試的內(nèi)容,可能發(fā)生錯(cuò)誤的邊界條件。
          下面是對(duì)Largest做的測試計(jì)劃:
          1、 數(shù)組元素的位置是否對(duì)最大值產(chǎn)生影響?
          l [7,8,9] – 9
          l [7,9,8] – 9
          l [9,8,7] – 9
          2、 如果有兩個(gè)相等的最大值,會(huì)出現(xiàn)什么情況呢?
          l [7,9,8,9] – 9
          3、如果數(shù)組中只有一個(gè)元素,結(jié)果會(huì)怎么樣?
          l [1] - 1
          4、 如果元素都是負(fù)數(shù)呢?
          l [-7,-8,-9] - -7
          完整的測試代碼應(yīng)該如下:
          public class LargestTest extends TestCase {
           
           public void testSimple(){
            assertEquals(9,Largest.largest(new int[]{7,8,9}));
           }
           public void testOrder(){
            assertEquals(9,Largest.largest(new int[]{7,9,8}));
            assertEquals(9,Largest.largest(new int[]{9,8,7}));
           }
           public void testDups(){
            assertEquals(9,Largest.largest(new int[]{7,9,8,9}));
           }
           public void testOne(){
            assertEquals(1,Largest.largest(new int[]{1}));
           }
           public void testNegative(){
             assertEquals(-7,Largest.largest(new int[]{-7,-8,-9}));
           }
          }
          當(dāng)然,你可以寫完一個(gè)測試方法就立即來運(yùn)行它。這次并沒有那么幸運(yùn)了,在運(yùn)行最后一個(gè)測試方法testNegative()時(shí)出現(xiàn)了錯(cuò)誤:
          junit.framework.AssertionFailedError: expected:<-7> but was:<0>
           at test.junit.LargestTest.testNegative(LargestTest.java:24)
          細(xì)心的你,也許在一開始就發(fā)現(xiàn)了Largest的這個(gè)Bug。原來我們的字段max初始化為0是不對(duì)的,應(yīng)該改為Integer.MIN_VALUE。
          由此我們可以想到,使用單元測試確實(shí)可以盡早的發(fā)現(xiàn)隱藏的BUG,上一篇我們也說過,越早發(fā)現(xiàn)BUG就能節(jié)省更多的時(shí)間,降低更多的風(fēng)險(xiǎn)。
          這是,我們的單元測試已經(jīng)完美結(jié)束了嗎?呵呵,也許你會(huì)想到,如果在largest()方法中傳入數(shù)組為空,又會(huì)怎么樣呢?這個(gè)問題留給我們的讀者思考吧。


          寫到這里,算是入門結(jié)束了吧!關(guān)于JUnit的詳細(xì)介紹,網(wǎng)上有非常多的文章,去google你可以找到一大堆。下面我提供幾個(gè)不錯(cuò)的單元測試網(wǎng)站,希望能對(duì)你有所幫助:
          51Testing-無憂軟件測試網(wǎng):http://www.51testing.com/
          測試時(shí)代:http://www.testage.net/
          UML軟件工程組織-軟件測試:http://www.uml.org.cn/Test/test.asp
          測試管理中心:http://www.testmanager.com.cn/
          軟件工程專家網(wǎng):http://www.51cmm.com/
          開放軟件測試研究:http://www.opentest.net/

          posted on 2005-04-11 14:46 閱讀(237) 評(píng)論(0)  編輯  收藏 所屬分類: 測試

          主站蜘蛛池模板: 虞城县| 海淀区| 中宁县| 临洮县| 高清| 莎车县| 海盐县| 如皋市| 灵寿县| 鄂尔多斯市| 陇西县| 长兴县| 泰安市| 南木林县| 泰宁县| 衡阳县| 田东县| 长沙县| 扶绥县| 宁明县| 新余市| 泌阳县| 繁峙县| 巴里| 新乐市| 尼玛县| 定远县| 甘肃省| 萍乡市| 红原县| 逊克县| 广宁县| 白河县| 上虞市| 衡阳县| 汶川县| 盐边县| 墨玉县| 晋中市| 河北省| 双鸭山市|