Sung in Blog

                     一些技術(shù)文章 & 一些生活雜碎
          JUnit是由 Erich Gamma 和 Kent Beck 編寫的一個(gè)回歸測(cè)試框架(regression testing framework),供Java開發(fā)人員編寫單元測(cè)試之用。

          1、概述
            Junit測(cè)試是程序員測(cè)試,即所謂白盒測(cè)試,因?yàn)槌绦騿T知道被測(cè)試的軟件如何(How)完成功能和完成什么樣(What)的功能。
            Junit本質(zhì)上是一套框架,即開發(fā)者制定了一套條條框框,遵循這此條條框框要求編寫測(cè)試代碼,如繼承某個(gè)類,實(shí)現(xiàn)某個(gè)接口,就可以用Junit進(jìn)行自動(dòng)測(cè)試了。
            由于Junit相對(duì)獨(dú)立于所編寫的代碼,可以測(cè)試代碼的編寫可以先于實(shí)現(xiàn)代碼的編寫,XP 中推崇的 test first design的實(shí)現(xiàn)有了現(xiàn)成的手段:用Junit寫測(cè)試代碼,寫實(shí)現(xiàn)代碼,運(yùn)行測(cè)試,測(cè)試失敗,修改實(shí)現(xiàn)代碼,再運(yùn)行測(cè)試,直到測(cè)試成功。以后對(duì)代碼的修改和優(yōu)化,運(yùn)行測(cè)試成功,則修改成功。
            Java 下的 team 開發(fā),采用 cvs(版本控制) + ant(項(xiàng)目管理) + junit(集成測(cè)試) 的模式時(shí),通過對(duì)ant的配置,可以很簡(jiǎn)單地實(shí)現(xiàn)測(cè)試自動(dòng)化。

            對(duì)不同性質(zhì)的被測(cè)對(duì)象,如Class,Jsp,Servlet,Ejb等,Junit有不同的使用技巧,以后慢慢地分別講敘。以下以Class測(cè)試為例講解,除非特殊說明。

          2、下載安裝

          • Junit主頁(yè)下載最新版本3.8.1程序包junit-3.8.1.zip
          • 用winzip或unzip將junit-3.8.1.zip解壓縮到某一目錄名為$JUNITHOME
          • 將junit.jar和$JUNITHOME/junit加入到CLASSPATH中,加入后者只因?yàn)闇y(cè)試?yán)淘谀莻€(gè)目錄下。
          • 注意不要將junit.jar放在jdk的extension目錄下
          • 運(yùn)行命令,結(jié)果如下圖。
            java junit.swingui.TestRunner junit.samples.AllTests

          junit-alltest.gif

          3、Junit架構(gòu)
            下面以Money這個(gè)類為例進(jìn)行說明。

          public class Money {
              private int fAmount;//余額
              private String fCurrency;//貨幣類型

              public Money(int amount, String currency) {
                  fAmount= amount;
                  fCurrency= currency;
              }

              public int amount() {
                  return fAmount;
              }

              public String currency() {
                  return fCurrency;
              }
              
              public Money add(Money m) {//加錢
                  return new Money(amount()+m.amount(), currency());
              }
              
              public boolean equals(Object anObject) {//判斷錢數(shù)是否相等
                  if (anObject instanceof Money) {
                      Money aMoney= (Money)anObject;
                      return aMoney.currency().equals(currency())
                          && amount() == aMoney.amount();
                  }
                  return false;
              }    
          }


            Junit本身是圍繞著兩個(gè)設(shè)計(jì)模式來(lái)設(shè)計(jì)的:命令模式集成模式.
          • 命令模式
              利用TestCase定義一個(gè)子類,在這個(gè)子類中生成一個(gè)被測(cè)試的對(duì)象,編寫代碼檢測(cè)某個(gè)方法被調(diào)用后對(duì)象的狀態(tài)與預(yù)期的狀態(tài)是否一致,進(jìn)而斷言程序代碼有沒有bug。
              當(dāng)這個(gè)子類要測(cè)試不只一個(gè)方法的實(shí)現(xiàn)代碼時(shí),可以先建立測(cè)試基礎(chǔ),讓這些測(cè)試在同一個(gè)基礎(chǔ)上運(yùn)行,一方面可以減少每個(gè)測(cè)試的初始化,而且可以測(cè)試這些不同方法之間的聯(lián)系。
              例如,我們要測(cè)試Money的Add方法,可以如下:
            public class MoneyTest extends TestCase { //TestCase的子類
                public void testAdd() { //把測(cè)試代碼放在testAdd中
                    Money m12CHF= new Money(12, "CHF");  //本行和下一行進(jìn)行一些初始化
                    Money m14CHF= new Money(14, "CHF");        
                    Money expected= new Money(26, "CHF");//預(yù)期的結(jié)果
                    Money result= m12CHF.add(m14CHF);    //運(yùn)行被測(cè)試的方法
                    Assert.assertTrue(expected.equals(result));     //判斷運(yùn)行結(jié)果是否與預(yù)期的相同
                }
            }

              如果測(cè)試一下equals方法,用類似的代碼,如下:
            public class MoneyTest extends TestCase { //TestCase的子類
                public void testEquals() { //把測(cè)試代碼放在testEquals中
                    Money m12CHF= new Money(12, "CHF"); //本行和下一行進(jìn)行一些初始化
                    Money m14CHF= new Money(14, "CHF");

                    Assert.assertTrue(!m12CHF.equals(null));//進(jìn)行不同情況的測(cè)試
                    Assert.assertEquals(m12CHF, m12CHF);
                    Assert.assertEquals(m12CHF, new Money(12, "CHF")); // (1)
                    Assert.assertTrue(!m12CHF.equals(m14CHF));
                }
            }


              當(dāng)要同時(shí)進(jìn)行測(cè)試Add和equals方法時(shí),可以將它們的各自的初始化工作,合并到一起進(jìn)行,形成測(cè)試基礎(chǔ),用setUp初始化,用tearDown清除。如下:
            public class MoneyTest extends TestCase {//TestCase的子類
                private Money f12CHF;//提取公用的對(duì)象
                private Money f14CHF;   

                protected void setUp() {//初始化公用對(duì)象
                    f12CHF= new Money(12, "CHF");
                    f14CHF= new Money(14, "CHF");
                }
                public void testEquals() {//測(cè)試equals方法的正確性
                    Assert.assertTrue(!f12CHF.equals(null));
                    Assert.assertEquals(f12CHF, f12CHF);
                    Assert.assertEquals(f12CHF, new Money(12, "CHF"));
                    Assert.assertTrue(!f12CHF.equals(f14CHF));
                }
                
                public void testSimpleAdd() {//測(cè)試add方法的正確性
                    Money expected= new Money(26, "CHF");
                    Money result= f12CHF.add(f14CHF);
                    Assert.assertTrue(expected.equals(result));
                }
            }


              將以上三個(gè)中的任一個(gè)TestCase子類代碼保存到名為MoneyTest.java的文件里,并在文件首行增加
            import junit.framework.*;
            ,都是可以運(yùn)行的。關(guān)于Junit運(yùn)行的問題很有意思,下面單獨(dú)說明。
              上面為解釋概念“測(cè)試基礎(chǔ)(fixture)”,引入了兩個(gè)對(duì)兩個(gè)方法的測(cè)試。命令模式與集成模式的本質(zhì)區(qū)別是,前者一次只運(yùn)行一個(gè)測(cè)試。
          • 集成模式
              利用TestSuite可以將一個(gè)TestCase子類中所有test***()方法包含進(jìn)來(lái)一起運(yùn)行,還可將TestSuite子類也包含進(jìn)來(lái),從而行成了一種等級(jí)關(guān)系。可以把TestSuite視為一個(gè)容器,可以盛放TestCase中的test***()方法,它自己也可以嵌套。這種體系架構(gòu),非常類似于現(xiàn)實(shí)中程序一步步開發(fā)一步步集成的現(xiàn)況。
              對(duì)上面的例子,有代碼如下:
            public class MoneyTest extends TestCase {//TestCase的子類
                ....
                public static Test suite() {//靜態(tài)Test
                    TestSuite suite= new TestSuite();//生成一個(gè)TestSuite
                    suite.addTest(new MoneyTest("testEquals")); //加入測(cè)試方法
                    suite.addTest(new MoneyTest("testSimpleAdd"));
                    return suite;
                }
            }

              從Junit2.0開始,有列簡(jiǎn)捷的方法:
            public class MoneyTest extends TestCase {//TestCase的子類
                ....
                public static Test suite() {靜態(tài)Test
                    return new TestSuite(MoneyTest.class); //以類為參數(shù)
                }
            }

              TestSuite見嵌套的例子,在后面應(yīng)用案例中有。
              

          4、測(cè)試代碼的運(yùn)行
            先說最常用的集成模式。
            測(cè)試代碼寫好以后,可以相應(yīng)的類中寫main方法,用java命令直接運(yùn)行;也可以不寫main方法,用Junit提供的運(yùn)行器運(yùn)行。Junit提供了textui,awtui和swingui三種運(yùn)行器。
            以前面第2步中的AllTests運(yùn)行為例,可有四種:

          java junit.textui.TestRunner junit.samples.AllTests
          java junit.awtui.TestRunner junit.samples.AllTests
          java junit.swingui.TestRunner junit.samples.AllTests
          java junit.samples.AllTests

            main方法中一般也都是簡(jiǎn)單地用Runner調(diào)用suite(),當(dāng)沒有main時(shí),TestRunner自己以運(yùn)行的類為參數(shù)生成了一個(gè)TestSuite.
            
            對(duì)于命令模式的運(yùn)行,有兩種方法。
          • 靜態(tài)方法
            TestCase test= new MoneyTest("simple add") {
            public void runTest() {
            testSimpleAdd();
            }
            };

          • 動(dòng)態(tài)方法
            TestCase test= new MoneyTest("testSimpleAdd");

            我試了一下,好象有問題,哪位朋友成功了,請(qǐng)指點(diǎn)我一下。確實(shí)可以。
          import junit.framework.*;

          public class MoneyTest extends TestCase {//TestCase的子類
              private Money f12CHF;//提取公用的對(duì)象
              private Money f14CHF;   
              public MoneyTest(String name){
                  super(name);
              }
              protected void setUp() {//初始化公用對(duì)象
                  f12CHF= new Money(12, "CHF");
                  f14CHF= new Money(14, "CHF");
              }
              public void testEquals() {//測(cè)試equals方法的正確性
                  Assert.assertTrue(!f12CHF.equals(null));
                  Assert.assertEquals(f12CHF, f12CHF);
                  Assert.assertEquals(f12CHF, new Money(12, "CHF"));
                  Assert.assertTrue(!f12CHF.equals(f14CHF));
              }
              
              public void testAdd() {//測(cè)試add方法的正確性
                  Money expected= new Money(26, "CHF");
                  Money result= f12CHF.add(f14CHF);
                  Assert.assertTrue(expected.equals(result));
              }
          //    public static void main(String[] args) {
          //        TestCase test=new MoneyTest("simple add") {
          //                public void runTest() {
          //                    testAdd();
          //                }
          //            };
          //        junit.textui.TestRunner.run(test);
          //    }
              public static void main(String[] args) {
                  TestCase test=new MoneyTest("testAdd");
                  junit.textui.TestRunner.run(test);
              }
          }


          再給一個(gè)靜態(tài)方法用集成測(cè)試的例子:
          public static Test suite() {
              TestSuite suite= new TestSuite();
              suite.addTest(
                  new testCar("getWheels") {
                      protected void runTest() { testGetWheels(); }
                  }
              );

              suite.addTest(
                  new testCar("getSeats") {
                      protected void runTest() { testGetSeats(); }
                  }
              );
              return suite;
          }

          5、應(yīng)用案例

          1. Junit Primer例程,運(yùn)行如下:
            java com.hedong.JunitLearning.Primer.ShoppingCartTest

          2. Ant+Junit+Mailto實(shí)現(xiàn)自動(dòng)編譯、調(diào)試并發(fā)送結(jié)果的build.xml
          3. JUnit實(shí)施,寫得很棒,理解也深刻。例程運(yùn)行如下:
            java com.hedong.JunitLearning.car.testCarNoJunit
            java junit.swingui.TestRunner com.hedong.JunitLearning.car.testCar

          4. Junit與log4j結(jié)合,阿菜的例程運(yùn)行:
            cd acai
            ant junit






          6、一些問題
            有人在實(shí)踐基礎(chǔ)上總結(jié)出一些非常有價(jià)值的使用技巧,我沒有經(jīng)過一一“測(cè)試”,暫列在此。
          1. 不要用TestCase的構(gòu)造函數(shù)初始化Fixture,而要用setUp()和tearDown()方法。
          2. 不要依賴或假定測(cè)試運(yùn)行的順序,因?yàn)镴Unit利用Vector保存測(cè)試方法。所以不同的平臺(tái)會(huì)按不同的順序從Vector中取出測(cè)試方法。不知3.8中是不是還是如此,不過它提供的例子有一個(gè)是指定用VectorSuite的,如果不指定呢?
          3. 避免編寫有副作用的TestCase。例如:如果隨后的測(cè)試依賴于某些特定的交易數(shù)據(jù),就不要提交交易數(shù)據(jù)。簡(jiǎn)單的回滾就可以了。
          4. 當(dāng)繼承一個(gè)測(cè)試類時(shí),記得調(diào)用父類的setUp()和tearDown()方法。
          5. 將測(cè)試代碼和工作代碼放在一起,一邊同步編譯和更新。(使用Ant中有支持junit的task.)
          6. 測(cè)試類和測(cè)試方法應(yīng)該有一致的命名方案。如在工作類名前加上test從而形成測(cè)試類名。
          7. 確保測(cè)試與時(shí)間無(wú)關(guān),不要依賴使用過期的數(shù)據(jù)進(jìn)行測(cè)試。導(dǎo)致在隨后的維護(hù)過程中很難重現(xiàn)測(cè)試。
          8. 如果你編寫的軟件面向國(guó)際市場(chǎng),編寫測(cè)試時(shí)要考慮國(guó)際化的因素。不要僅用母語(yǔ)的Locale進(jìn)行測(cè)試。
          9. 盡可能地利用JUnit提供地assert/fail方法以及異常處理的方法,可以使代碼更為簡(jiǎn)潔。
          10. 測(cè)試要盡可能地小,執(zhí)行速度快。
          11. 把測(cè)試程序建立在與被測(cè)對(duì)象相同的包中
          12. 在你的原始代碼目錄中避免測(cè)試碼出現(xiàn),可在一個(gè)源碼鏡像目錄中放測(cè)試碼
          13. 在自己的應(yīng)用程序包中包含一個(gè)TestSuite測(cè)試類



          7、相關(guān)資源下載
          以下jar包,我只是做了打包、編譯和調(diào)試的工作,供下載學(xué)習(xí)之用,相關(guān)的權(quán)利屬于原作者。
          1. 可運(yùn)行例程.jar
          2. Build.xml
          3. 阿菜的例程
          4. Junit API 漢譯(pdf)

          8、未完成的任務(wù)

          1. httpunit
          2. cactus
          3. 將Junit用鏈接池測(cè)試

          主要參考文獻(xiàn):

          1. JUnit入門
            http://www.dotspace.twmail.org/Test/JUnit_Primer.htm
          2. 怎樣使用Junit Framework進(jìn)行單元測(cè)試的編寫
            http://www.chinaunix.net/bbsjh/14/546.html
          3. Ant+Junit+Log4J+CVS進(jìn)行XP模式開發(fā)的建立
            http://ejb.cn/modules/tutorials/printpage.php?tid=4
          4. 用HttpUnit測(cè)試Web應(yīng)用程序
            http://www.zdnet.com.cn/developer/code/story/0,2000081534,39033726,00.htm
          5. 有沒有用過Cactus的,Web層的測(cè)試是Cactus還是JUnit?
            http://www.jdon.com/jive/thread.jsp?forum=16&thread=9156
          6. Ant+junit的測(cè)試自動(dòng)化 biggie(原作)
            http://www.csdn.net/Develop/article/19%5C19748.shtm
          7. JUnit實(shí)施
            http://www.neweasier.com/article/2002-08-07/1028723459.html
          8. JUnitTest Infected: Programmers Love Writing Tests
            http://junit.sourceforge.net/doc/testinfected/testing.htm
          9. JUnit Cookbook
            http://junit.sourceforge.net/doc/cookbook/cookbook.htm
          10. JUnit Primer
            http://www.itu.dk/~lthorup/JUnitPrimer.html
          11. IBM DevelopWorks
            http://www-106.ibm.com/search/searchResults.jsp?query=junit&searchScope=dW&
            searchType=1&searchSite=dWChina&pageLang=zh&langEncoding=gb2312&Search.x=0&
            Search.y=0&Search=Search
          posted on 2005-10-16 13:04 Sung 閱讀(209) 評(píng)論(0)  編輯  收藏 所屬分類: Java
          主站蜘蛛池模板: 阳新县| 宜兰县| 石屏县| 九寨沟县| 南雄市| 永定县| 井冈山市| 东乡族自治县| 遂溪县| 万宁市| 杭州市| 肇东市| 楚雄市| 桂阳县| 印江| 江永县| 屯留县| 兴文县| 黄大仙区| 田阳县| 林周县| 德州市| 长宁县| 南开区| 鄂伦春自治旗| 沙雅县| 兰溪市| 双峰县| 肇庆市| 康马县| 合江县| 宁南县| 广安市| 自贡市| 威远县| 诸暨市| 沁水县| 昌都县| 寿宁县| 门头沟区| 谢通门县|