隨筆 - 4  文章 - 16  trackbacks - 0
          <2008年11月>
          2627282930311
          2345678
          9101112131415
          16171819202122
          23242526272829
          30123456

          常用鏈接

          留言簿(3)

          隨筆分類

          隨筆檔案

          文章分類

          新聞檔案

          好友

          最新隨筆

          搜索

          •  

          積分與排名

          • 積分 - 11258
          • 排名 - 2285

          最新評(píng)論

          閱讀排行榜

          評(píng)論排行榜

          本節(jié)是單元測(cè)試系列的第二篇。重點(diǎn)講解如何使用Mock/Stub和依賴注入技術(shù)進(jìn)行單元測(cè)試。關(guān)于工具JUnit等則不做累贅介紹。 希望通過本章能夠幫助大家開始單元測(cè)試的有益實(shí)踐,與大家共勉!

          單元測(cè)試(技能篇)

          一、Stub技術(shù)

          這是最為古老的一種測(cè)試技能。通過類層次上的替換實(shí)現(xiàn)了對(duì)待測(cè)環(huán)境的模擬。

          實(shí)現(xiàn)的時(shí)候有兩種途徑:

          1、重寫實(shí)際類,在測(cè)試時(shí),先于實(shí)際類加載,即覆蓋。如:我們?cè)?/span>unittest/stub文件夾下針對(duì)于每一個(gè)重寫類都有相同的包結(jié)構(gòu)和類名:


          在類路徑中優(yōu)先加載:


          2、在實(shí)際代碼中添加判斷。比如,如果當(dāng)前是測(cè)試環(huán)境if(isUT)執(zhí)行XX操作,截?cái)嗾嬲枰龅氖隆?/span>

              publicvoid sendCommand(int cmdCode)

              {

                 if(isUT())

                 {

                     //...

                 }

                 else

                 {

                     //...

                 }

              }

          Stub技術(shù)的問題就在于我們?cè)谥貙戇@些類的時(shí)候,不僅僅要關(guān)注接口,還要關(guān)注其內(nèi)部邏輯。如果你只是簡(jiǎn)單的返回一個(gè)固定的響應(yīng),會(huì)很簡(jiǎn)單。但是對(duì)于每一次運(yùn)行需要根據(jù)不同的輸入返回不同的輸出時(shí)方法內(nèi)部的處理就會(huì)復(fù)雜的多。

          由于實(shí)現(xiàn)的難度,所以,使用時(shí)就要注意:有高價(jià)值、重用度高、數(shù)量少。這就是說(shuō),重寫一個(gè)類,就可以有一大批類可以用。

          二、Mock技術(shù)

          Mock是目前單元測(cè)試中最常用的。用來(lái)在對(duì)象層次上實(shí)現(xiàn)細(xì)類度替換十分方便。

          當(dāng)我們?cè)跍y(cè)試中,需要其它類/接口的一個(gè)方法時(shí),我們可以通過繼承/實(shí)現(xiàn)其一個(gè)子類對(duì)象來(lái)替換實(shí)際對(duì)象。在Mock子類中將需要的方法直接返回需要的結(jié)果就行了。

              privateclass Mock_QueryCtrl extends QueryCtrl

              {

                 public List queryNEList()

                 {

                     List neList = new ArrayList();

                     //直接填充并返回你需要的數(shù)據(jù)...

                     return neList;

                 }

              }

          同樣,我們也可以通過測(cè)試待測(cè)類的子類來(lái)測(cè)試待測(cè)類。這對(duì)于被測(cè)方法使用了自身類的方法時(shí)很適用。

          三、依賴注入

          單元測(cè)試的一個(gè)關(guān)鍵就是替換。類層次上的替換,通過在類路徑中提前加載就可以實(shí)現(xiàn)。而在對(duì)象層次上,java的反射機(jī)制提供了很好的幫助。

          1.獲取/注入私有屬性

          2.執(zhí)行私有方法

          附:注入私有屬性的實(shí)現(xiàn):

              publicvoid setFieldObject(Object instance, String fieldName, Object value)

                     throws IllegalArgumentException, IllegalAccessException,

                     NoSuchFieldException {

                 Field field = null;

                 Class c = instance.getClass();

                 do {

                     try

                     {

                        field = c.getDeclaredField(fieldName);

                     } catch (SecurityException e)

                     {

                        e.printStackTrace();

                     } catch (NoSuchFieldException e)

                     {

                        c = c.getSuperclass();

                     }

                 }

                 while (c.getName() != "java.lang.Object" && field == null);

                 if (field != null)

                 {

                     field.setAccessible(true);

                     field.set(instance, value);

                 }

                 else

                 {

                     thrownew NoSuchFieldException(fieldName);

                 }

              }

          注:這是一個(gè)簡(jiǎn)單實(shí)現(xiàn),實(shí)際中需要優(yōu)化。

          四、實(shí)例:

          下例演示了如何測(cè)試類NEListTableShowNETable()方法。其中注意的是,方法中調(diào)用了類QueryCtrlqueryNEList()方法。

          待測(cè)類:

          publicclass NEListTable

          {

              QueryCtrl ctrl = null;

             

              publicvoid ShowNETable()

              {

                 List neList = ctrl.queryNEList();

                

                 for(int i = 0;i<neList.size();i++)

                 {

                     //neList轉(zhuǎn)換為表格行

                 }

                

                 //顯示表格...

              }

          }

          publicclass QueryCtrl {

              public List queryNEList()

          {

                 returnnull;

              }

          }

          測(cè)試類:

          public class TestNEListTable extends TestCase

          {

              private NEListTable table = null;

              private TestHelper helper = null;

              public void testShowNETable()

              {

                 Mock_QueryCtrl ctrl = new Mock_QueryCtrl();

                 helper.setObjectField(table,"ctrl",ctrl);//Mock對(duì)象注入table

                 table.ShowNETable();

                 assertTrue(table.getRowCount()>0);

              }

             

              private class Mock_QueryCtrl extends QueryCtrl

              {

                 public List queryNEList()

                 {

                     List neList = new ArrayList();

                     //返回你需要的數(shù)據(jù)...

                     return neList;

                 }

              }

          }

          posted on 2008-07-10 22:14 wukaichun 閱讀(2033) 評(píng)論(4)  編輯  收藏 所屬分類: test

          FeedBack:
          # re: 單元測(cè)試(技能篇) 2008-07-21 21:34 切爾斯基
          Mock不是你說(shuō)的這個(gè)樣子吧, 你這里說(shuō)的mock其實(shí)是stub  回復(fù)  更多評(píng)論
            
          # re: 單元測(cè)試(技能篇) 2008-07-21 23:19 wukaichun
          @切爾斯基

          非常感謝.不知您的見解是什么?
          我的理解是:Stub的替換發(fā)生在類層面,而Mock則是對(duì)象層面. 前者粒度較大,更適合功能測(cè)試級(jí);后者粒度相對(duì)較小,對(duì)于方法級(jí)的單元測(cè)試比較有效.
            回復(fù)  更多評(píng)論
            
          # re: 單元測(cè)試(技能篇) 2008-07-28 22:42 Growing
          沒錯(cuò),同意"切爾斯基"的說(shuō)法,樓主的Mock其實(shí)就是Stub,Mock與Stub的區(qū)別我認(rèn)為是在于:Mock不需要像Stub那樣增加類。而是通過目前使用較多的EasyMock和JMock等框架,通過動(dòng)態(tài)代理等方式生成代理類,使用這個(gè)代理類代替被測(cè)試類的依賴對(duì)象。保證測(cè)試的獨(dú)立、隔離。  回復(fù)  更多評(píng)論
            
          # re: 單元測(cè)試(技能篇) 2008-11-19 14:54 網(wǎng)絡(luò)監(jiān)控軟件
          能不能介紹一些C++的單元測(cè)試框架。  回復(fù)  更多評(píng)論
            

          只有注冊(cè)用戶登錄后才能發(fā)表評(píng)論。


          網(wǎng)站導(dǎo)航:
           
          主站蜘蛛池模板: 柘城县| 三门峡市| 龙井市| 连州市| 建湖县| 酒泉市| 从化市| 福安市| 两当县| 精河县| 稷山县| 田阳县| 印江| 东乡县| 礼泉县| 高台县| 同江市| 祁门县| 凤山县| 封丘县| 平和县| 金乡县| 方城县| 长治县| 镶黄旗| 泸西县| 皮山县| 石阡县| 崇州市| 平乡县| 封开县| 伽师县| 含山县| 河源市| 阿拉善盟| 德清县| 新丰县| 建水县| 白水县| 博客| 滦南县|