寫好單元測(cè)試的7個(gè)要點(diǎn)

寫好單元測(cè)試的7個(gè)要點(diǎn)

測(cè)試是開(kāi)發(fā)的一個(gè)非常重要的方面粱侣,可以在很大程度上決定應(yīng)用程序的命運(yùn)券坞。好的測(cè)試可以在早期捕獲應(yīng)用程序終止問(wèn)題琅锻,但是糟糕的測(cè)試總是會(huì)導(dǎo)致失敗和停止帮掉。

雖然有三種主要的軟件測(cè)試類型:?jiǎn)卧獪y(cè)試、功能測(cè)試和集成測(cè)試晕鹊,但在本文中松却,我將討論開(kāi)發(fā)人員級(jí)的單元測(cè)試。在我深入研究細(xì)節(jié)之前溅话,讓我們從高層次上回顧一下每種類型的測(cè)試需要什么。

軟件測(cè)試的類型

單元測(cè)試 用于測(cè)試單個(gè)代碼組件歌焦,并確保代碼按預(yù)期方式工作飞几。單元測(cè)試由開(kāi)發(fā)人員編寫和執(zhí)行。大多數(shù)情況下独撇,會(huì)使用JUnit或TestNG這樣的測(cè)試框架屑墨。測(cè)試用例通常在方法級(jí)別編寫,并通過(guò)自動(dòng)化執(zhí)行纷铣。

集成測(cè)試 檢查整個(gè)系統(tǒng)是否工作正常卵史。集成測(cè)試也是由開(kāi)發(fā)人員完成的,但它不是測(cè)試單個(gè)組件搜立,而是旨在跨組件進(jìn)行測(cè)試以躯。系統(tǒng)由許多單獨(dú)的組件組成,如代碼啄踊、數(shù)據(jù)庫(kù)忧设、Web服務(wù)器等。集成測(cè)試能夠發(fā)現(xiàn)組件的連接颠通、網(wǎng)絡(luò)訪問(wèn)址晕、數(shù)據(jù)庫(kù)問(wèn)題等問(wèn)題。

功能測(cè)試 通過(guò)將給定輸入的結(jié)果與規(guī)范進(jìn)行比較來(lái)檢查每個(gè)特性是否正確實(shí)現(xiàn)顿锰。通常谨垃,這不是在開(kāi)發(fā)人員級(jí)別完成的启搂。功能測(cè)試由單獨(dú)的測(cè)試團(tuán)隊(duì)執(zhí)行。根據(jù)規(guī)范編寫測(cè)試用例刘陶,并將實(shí)際結(jié)果與預(yù)期結(jié)果進(jìn)行比較胳赌。有幾種工具可用于自動(dòng)化功能測(cè)試,如Selenium和qtp易核。

TIPS

1. 使用單元測(cè)試框架

Java提供了用于單元測(cè)試的若干框架匈织。testng和junit是最流行的測(cè)試框架。JUnit和TESTNG的一些重要特性:

  • 易于安裝和運(yùn)行牡直。支持批注缀匕。
  • 允許忽略或分組某些測(cè)試并一起執(zhí)行 。
  • 支持參數(shù)化測(cè)試碰逸,即通過(guò)在運(yùn)行時(shí)指定不同的值來(lái)運(yùn)行單元測(cè)試
  • 通過(guò)與Ant乡小、Maven和Gradle等構(gòu)建工具集成,支持自動(dòng)測(cè)試執(zhí)行饵史。

EasyMock是一個(gè)模擬框架满钟,它是對(duì)諸如JUnit和TestNG這樣的單元測(cè)試框架的補(bǔ)充。easymock本身不是一個(gè)完整的框架胳喷。它只是增加了創(chuàng)建模擬對(duì)象以方便測(cè)試的能力湃番。例如,我們要測(cè)試的方法可以調(diào)用從數(shù)據(jù)庫(kù)獲取數(shù)據(jù)的DAO類吭露。在這種情況下吠撮,easymock可以用來(lái)創(chuàng)建返回硬編碼數(shù)據(jù)的mockdao。這使得我們可以輕松地測(cè)試我們想要的方法讲竿,而不必為數(shù)據(jù)庫(kù)訪問(wèn)而煩惱泥兰。

2. 強(qiáng)烈建議使用測(cè)試驅(qū)動(dòng)開(kāi)發(fā)!

測(cè)試驅(qū)動(dòng)開(kāi)發(fā)(TDD)是一個(gè)軟件開(kāi)發(fā)過(guò)程题禀,在這個(gè)過(guò)程中鞋诗,在任何編碼開(kāi)始之前,測(cè)試都是根據(jù)需求編寫的迈嘹。由于還沒(méi)有代碼削彬,測(cè)試最初將失敗。然后寫入最小數(shù)量的代碼以通過(guò)測(cè)試江锨。然后重構(gòu)代碼吃警,逐步優(yōu)化。

其目標(biāo)是編寫涵蓋所有需求的測(cè)試啄育,而不是簡(jiǎn)單地先編寫甚至可能不滿足需求的代碼酌心。TDD非常好,因?yàn)樗帉懥艘子诰S護(hù)的簡(jiǎn)單模塊化代碼挑豌“踩總體開(kāi)發(fā)速度加快墩崩,缺陷容易識(shí)別。此外侯勉,單元測(cè)試是作為TDD方法的副產(chǎn)品創(chuàng)建的鹦筹。

但是,TDD可能不適用于所有情況址貌。在設(shè)計(jì)復(fù)雜的項(xiàng)目中铐拐,專注于最簡(jiǎn)單的設(shè)計(jì)以通過(guò)測(cè)試用例,而不提前考慮可能會(huì)導(dǎo)致巨大的代碼更改练对。此外遍蟋,對(duì)于與遺留系統(tǒng)、GUI應(yīng)用程序或與數(shù)據(jù)庫(kù)一起工作的應(yīng)用程序交互的系統(tǒng)螟凭,TDD方法也很難使用虚青。此外,測(cè)試需要隨著代碼的變化而更新螺男。

因此棒厘,在決定采用TDD方法之前,應(yīng)牢記上述因素下隧,并根據(jù)項(xiàng)目的性質(zhì)鑒別是否使用奢人。

3. 評(píng)估代碼覆蓋率

代碼覆蓋率度量在運(yùn)行單元測(cè)試時(shí)執(zhí)行代碼的百分比。通常淆院,具有高覆蓋率的代碼包含未檢測(cè)到的錯(cuò)誤的可能性會(huì)降低达传,因?yàn)樵跍y(cè)試過(guò)程中執(zhí)行了更多的源代碼。衡量代碼覆蓋率的一些最佳實(shí)踐包括:

  • 使用代碼覆蓋工具迫筑,如Clover、Corbetura宗弯、Jacoco或Sonar脯燃。
  • 使用工具可以提高測(cè)試質(zhì)量,因?yàn)檫@些工具可以指出代碼中未測(cè)試的部分蒙保,從而允許您開(kāi)發(fā)額外的測(cè)試來(lái)覆蓋這些區(qū)域辕棚。
  • 每當(dāng)編寫新功能時(shí),立即編寫要覆蓋的新測(cè)試邓厕。
  • 確保有覆蓋代碼所有分支的測(cè)試用例逝嚎,即if/else語(yǔ)句。

高代碼覆蓋率不能保證測(cè)試是完美的详恼,所以要小心补君!下面的concat方法接受一個(gè)布爾值作為輸入,并且只在布爾值為true時(shí)附加傳入的兩個(gè)字符串:

     public String concat(boolean append, String a,String b) {
        String result = null;
        If (append) {
            result = a + b;
                            }
        return result.toLowerCase();
    }

以下是上述方法的測(cè)試用例:

        @Test
        public void testStringUtil() {
         String result = stringUtil.concat(true, "Hello ", "World");
         System.out.println("Result is "+result);
        }

在這種情況下昧互,測(cè)試的執(zhí)行值為true挽铁。當(dāng)測(cè)試執(zhí)行時(shí)伟桅,它將通過(guò)。當(dāng)代碼覆蓋率工具運(yùn)行時(shí)叽掘,它將在執(zhí)行concat方法中的所有代碼時(shí)顯示100%的代碼覆蓋率楣铁。但是,如果使用值false執(zhí)行測(cè)試更扁,則將引發(fā)NullPointerException盖腕。因此,100%的代碼覆蓋率并不能真正表明測(cè)試是否覆蓋了所有場(chǎng)景浓镜,并且測(cè)試是否良好溃列。

4. 盡可能將測(cè)試數(shù)據(jù)外置化

在JUnit4之前,運(yùn)行測(cè)試用例的數(shù)據(jù)必須硬編碼到測(cè)試用例中竖哩。這創(chuàng)建了一個(gè)限制哭廉,為了用不同的數(shù)據(jù)運(yùn)行測(cè)試,必須修改測(cè)試用例代碼相叁。然而遵绰,junit4和testng支持將測(cè)試數(shù)據(jù)外部化,以便可以針對(duì)不同的數(shù)據(jù)集運(yùn)行測(cè)試用例增淹,而不必更改源代碼椿访。

下面的MathChecker類具有檢查數(shù)字是否為奇數(shù)的方法:

   public class MathChecker {
        public Boolean isOdd(int n) {
            if (n%2 != 0) {
                return true;
            } else {
                return false;
                                         }
        }
    }

TestNG
以下是MathChecker類使用testNg的測(cè)試用例:

        public class MathCheckerTest {
        private MathChecker checker;
        @BeforeMethod
        public void beforeMethod() {
          checker = new MathChecker();
        }
        @Test
        @Parameters("num")
        public void isOdd(int num) { 
          System.out.println("Running test for "+num);
          Boolean result = checker.isOdd(num);
          Assert.assertEquals(result, new Boolean(true));
        }
    }

下面是testng.xml(testng的配置文件),它具有要為其執(zhí)行測(cè)試的數(shù)據(jù):

   <?xml version="1.0" encoding="UTF-8"?>
    <suite name="ParameterExampleSuite" parallel="false">
    <test name="MathCheckerTest">
    <classes>
      <parameter name="num" value="3"></parameter>
      <class name="com.stormpath.demo.MathCheckerTest"/>
    </classes>
     </test>
     <test name="MathCheckerTest1">
    <classes>
      <parameter name="num" value="7"></parameter>
      <class name="com.stormpath.demo.MathCheckerTest"/>
    </classes>
     </test>
    </suite>

正如你看到的虑润,在這種情況下成玫,測(cè)試將執(zhí)行兩次,分別針對(duì)值3和7執(zhí)行一次拳喻。除了通過(guò)XML配置文件指定測(cè)試數(shù)據(jù)外哭当,還可以通過(guò)DataProvider注釋在類中提供測(cè)試數(shù)據(jù)。

JUnit
與testng類似冗澈,測(cè)試數(shù)據(jù)也可以為junit外部化钦勘。以下是上述同一MathChecker類的JUnit測(cè)試用例:

    @RunWith(Parameterized.class)
    public class MathCheckerTest {
     private int inputNumber;
     private Boolean expected;
     private MathChecker mathChecker;
     @Before
     public void setup(){
         mathChecker = new MathChecker();
     }
        // Inject via constructor
     public MathCheckerTest(int inputNumber, Boolean expected) {
         this.inputNumber = inputNumber;
         this.expected = expected;
     }
     @Parameterized.Parameters
     public static Collection<Object[]> getTestData() {
         return Arrays.asList(new Object[][]{
                 {1, true},
                 {2, false},
                 {3, true},
                 {4, false},
                 {5, true}
         });
     }
     @Test
     public void testisOdd() {
         System.out.println("Running test for:"+inputNumber);
         assertEquals(mathChecker.isOdd(inputNumber), expected);
     }
 }

可以看到,要為其執(zhí)行測(cè)試的測(cè)試數(shù)據(jù)是由gettestdata()方法指定的亚亲。這種方法可以很容易地修改為從外部文件讀取數(shù)據(jù)彻采,而不是使用硬編碼數(shù)據(jù)。

5. 使用斷言而不是打印語(yǔ)句

許多新開(kāi)發(fā)人員習(xí)慣于在每行代碼后編寫System.out.println語(yǔ)句捌归,以驗(yàn)證代碼是否正確執(zhí)行肛响。這種實(shí)踐通常擴(kuò)展到單元測(cè)試,導(dǎo)致測(cè)試代碼混亂惜索。除了混亂之外特笋,這還需要開(kāi)發(fā)人員手動(dòng)干預(yù)以驗(yàn)證控制臺(tái)上打印的輸出,以檢查測(cè)試是否成功運(yùn)行门扇。更好的方法是使用自動(dòng)指示測(cè)試結(jié)果的斷言雹有。

以下StringUtil類是一個(gè)簡(jiǎn)單類偿渡,其中一個(gè)方法連接兩個(gè)輸入字符串并返回結(jié)果:

    public class StringUtil {
        public String concat(String a,String b) {
            return a + b;
        }
    }

The following are two unit tests for the method above:

    @Test
    public void testStringUtil_Bad() {
         String result = stringUtil.concat("Hello ", "World");
         System.out.println("Result is "+result);
    }
    @Test
    public void testStringUtil_Good() {
         String result = stringUtil.concat("Hello ", "World");
         assertEquals("Hello World", result);
    }  

teststringutil \_bad將始終通過(guò),因?yàn)樗鼪](méi)有斷言霸奕。開(kāi)發(fā)人員需要在控制臺(tái)手動(dòng)驗(yàn)證測(cè)試的輸出溜宽。如果方法返回錯(cuò)誤的結(jié)果并且不需要開(kāi)發(fā)人員干預(yù),則teststringutil \_good將失敗质帅。

6. 生成具有確定性結(jié)果的測(cè)試

有些方法沒(méi)有確定的結(jié)果适揉,即該方法的輸出不是預(yù)先知道的,并且每次都可能變化煤惩。例如嫉嘀,考慮具有復(fù)雜函數(shù)的以下代碼和計(jì)算執(zhí)行復(fù)雜函數(shù)所需時(shí)間(毫秒)的方法:

    public class DemoLogic {
    private void veryComplexFunction(){
        //This is a complex function that has a lot of database access and is time consuming
        //To demo this method, I am going to add a Thread.sleep for a random number of milliseconds
        try {
            int time = (int) (Math.random()*100);
            Thread.sleep(time);
        } catch (InterruptedException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
    }
    public long calculateTime(){
        long time = 0;
        long before = System.currentTimeMillis();
        veryComplexFunction();
        long after = System.currentTimeMillis();
        time = after - before;
        return time;
    }
    }

在這種情況下,每次執(zhí)行calculateTime方法時(shí)魄揉,它將返回不同的值剪侮。為這個(gè)方法編寫測(cè)試用例沒(méi)有任何用處,因?yàn)榉椒ǖ妮敵鍪强勺兊穆逋恕R虼税旮瑴y(cè)試方法將無(wú)法驗(yàn)證任何特定執(zhí)行的輸出。

7. 測(cè)試錯(cuò)誤情景和邊界情景兵怯,以及正常情景

通常彩匕,開(kāi)發(fā)人員花費(fèi)大量的時(shí)間和精力編寫測(cè)試用例,以確保應(yīng)用程序按預(yù)期工作媒区。然而驼仪,測(cè)試錯(cuò)誤的測(cè)試用例也是很重要的。否定測(cè)試用例是測(cè)試系統(tǒng)是否可以處理無(wú)效數(shù)據(jù)的測(cè)試用例袜漩。例如绪爸,考慮一個(gè)簡(jiǎn)單的函數(shù),它讀取由用戶鍵入的長(zhǎng)度為8的字母數(shù)字值宙攻。除了字母數(shù)字值之外毡泻,還應(yīng)測(cè)試以下異常情景測(cè)試用例:

  • 用戶指定非字母數(shù)字值,如特殊字符
  • 用戶指定空值粘优。
  • 用戶指定的值大于或小于8個(gè)字符。

類似地呻顽,邊界測(cè)試用例測(cè)試系統(tǒng)是否能很好地處理極端值雹顺。例如,如果希望用戶輸入1到100之間的數(shù)值廊遍,1和100是邊界值嬉愧,測(cè)試系統(tǒng)中這些值非常重要。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末喉前,一起剝皮案震驚了整個(gè)濱河市没酣,隨后出現(xiàn)的幾起案子王财,更是在濱河造成了極大的恐慌,老刑警劉巖裕便,帶你破解...
    沈念sama閱讀 216,692評(píng)論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件绒净,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡偿衰,警方通過(guò)查閱死者的電腦和手機(jī)挂疆,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,482評(píng)論 3 392
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)下翎,“玉大人缤言,你說(shuō)我怎么就攤上這事∈邮拢” “怎么了胆萧?”我有些...
    開(kāi)封第一講書(shū)人閱讀 162,995評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)俐东。 經(jīng)常有香客問(wèn)我跌穗,道長(zhǎng),這世上最難降的妖魔是什么犬性? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,223評(píng)論 1 292
  • 正文 為了忘掉前任瞻离,我火速辦了婚禮,結(jié)果婚禮上乒裆,老公的妹妹穿的比我還像新娘套利。我一直安慰自己,他們只是感情好鹤耍,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,245評(píng)論 6 388
  • 文/花漫 我一把揭開(kāi)白布肉迫。 她就那樣靜靜地躺著,像睡著了一般稿黄。 火紅的嫁衣襯著肌膚如雪喊衫。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 51,208評(píng)論 1 299
  • 那天杆怕,我揣著相機(jī)與錄音族购,去河邊找鬼。 笑死陵珍,一個(gè)胖子當(dāng)著我的面吹牛寝杖,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播互纯,決...
    沈念sama閱讀 40,091評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼瑟幕,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起只盹,我...
    開(kāi)封第一講書(shū)人閱讀 38,929評(píng)論 0 274
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤辣往,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后殖卑,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體站削,經(jīng)...
    沈念sama閱讀 45,346評(píng)論 1 311
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,570評(píng)論 2 333
  • 正文 我和宋清朗相戀三年懦鼠,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了钻哩。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,739評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡肛冶,死狀恐怖街氢,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情睦袖,我是刑警寧澤珊肃,帶...
    沈念sama閱讀 35,437評(píng)論 5 344
  • 正文 年R本政府宣布,位于F島的核電站馅笙,受9級(jí)特大地震影響伦乔,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜董习,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,037評(píng)論 3 326
  • 文/蒙蒙 一烈和、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧皿淋,春花似錦招刹、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,677評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至哑舒,卻和暖如春妇拯,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背洗鸵。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 32,833評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工越锈, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人膘滨。 一個(gè)月前我還...
    沈念sama閱讀 47,760評(píng)論 2 369
  • 正文 我出身青樓瞪浸,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親吏祸。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,647評(píng)論 2 354

推薦閱讀更多精彩內(nèi)容

  • 測(cè)試是開(kāi)發(fā)的一個(gè)非常重要的方面,可以在很大程度上決定一個(gè)應(yīng)用程序的命運(yùn)贡翘。良好的測(cè)試可以在早期捕獲導(dǎo)致應(yīng)用程序崩潰的...
    java部落閱讀 867評(píng)論 0 6
  • 測(cè)試是開(kāi)發(fā)的一個(gè)非常重要的方面蹈矮,可以在很大程度上決定一個(gè)應(yīng)用程序的命運(yùn)。良好的測(cè)試可以在早期捕獲導(dǎo)致應(yīng)用程序崩潰的...
    java部落閱讀 222評(píng)論 0 0
  • 感謝原作者的奉獻(xiàn)鸣驱,原作者博客地址:http://blog.csdn.net/zhu_ai_xin_520/arti...
    狼孩閱讀 14,042評(píng)論 1 35
  • 說(shuō)起軟件測(cè)試四個(gè)字泛鸟,想必大家腦海中浮現(xiàn)的有集成測(cè)試、系統(tǒng)測(cè)試踊东、黑盒測(cè)試北滥、白盒測(cè)試等,可能就是沒(méi)想到會(huì)有單元測(cè)試闸翅。 ...
    斜杠時(shí)光閱讀 3,248評(píng)論 0 6
  • 說(shuō)起軟件測(cè)試四個(gè)字再芋,想必大家腦海中浮現(xiàn)的有集成測(cè)試、系統(tǒng)測(cè)試坚冀、黑盒測(cè)試济赎、白盒測(cè)試等,可能就是沒(méi)想到會(huì)有單元測(cè)試记某。 ...
    hahaoop閱讀 421評(píng)論 0 7