青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

CppExplore

一切像霧像雨又像風

  C++博客 :: 首頁 :: 新隨筆 :: 聯系 :: 聚合  :: 管理 ::
  29 隨筆 :: 0 文章 :: 280 評論 :: 0 Trackbacks
作者:CppExplore  http://www.shnenglu.com/CppExplore/http://blog.csdn.net/cppexplore同步發布
 

1 單元測試對象概述

各個對象組織結構如下:

 

 

職責簡述如下:

對象

職責

TestAssert

測試斷言:判定測試結果是否正確,一般類似斷言表達。

TestCase

測試用例:多個測試斷言組成一個測試用例。測試對象為一個類中的一個具體方法。

TestSuite

測試套件:多個測試用例組成一個測試套件。測試對象為一個類。

MainTestSuite

主測試套件:單元測試運行主程序入口。測試用例也可繞過測試套件,直接包含在主測試套件中。

TestFixture

測試夾具:用于測試前的初始化操作以及測試后的清理操作,一般包括準備測試的前置條件/測試對象的狀態設置等。


 

2 單元測試框架選型

當前存在很多流行的單元測試框架:衍生自JUnitCppUnit,以及簡化版本的CppUnitLite,Boost.Test測試框架,Google Test測試框架等。每個測試框架都很完善,都可勝任單元測試任務。

       從使用簡單性考慮,依次是Boost.Test>>Google Test>>CppUnitLite>>CppUnit.

本文選擇Boost.Test的單元測試框架講解。對Google Test感興趣的可參看http://www.cnblogs.com/coderzh/archive/2009/03/31/1426758.html。
 

Boost.Test UTF(Unit Test Framework)

3.1 LogLevel

講解TestAssert前,先說下Boost測試框架的日志級別,有以下9個級別:

級別名稱

說明

all / success

報告包括成功測試通知的所有日志信息

test_suite

顯示測試套件信息

message

顯示用戶信息

warning

報告用戶發出的警告

error

報告所有錯誤情況

cpp_exception

報告未捕獲的 C++ 異常

system_error

報告系統引起的非致命錯誤 (例如,超時或浮點數異常)

fatal

用戶或系統引起的致命錯誤 (例如,內存訪問越界)

nothing

不報告任何信息

生成測試可執行程序后,可以通過指定--log_level參數指定日志接別。比如,最后可執行程序為testmini,執行./testmini --log_level=warning指定在warning級別運行,默認執行在error級別。

請特別關注級別中黑體部分,TestAssert中將使用到。

3.2 TestAssert

Boost.Test中測試斷言包含如下3大類:

類別

功能

說明

WARN

打印warning日志,不增加失敗引用計數,繼續執行程序

檢驗不太重要但是正確的方面

CHECK

打印error日志,增加失敗引用計數,繼續執行程序

實現 assertions

REQUIRE

增加失敗應用計數,中斷程序的運行

失敗就不應該讓程序繼續運行則使用

 

Boost.Test中詳細測試斷言包含如下幾種:

TestAssert類型

說明

舉例

BOOST_WARN

WARN型預言檢測

BOOST_WARN(2+2==4); 

BOOST_CHECK

CHECK型預言檢測

BOOST_CHECK(2+2==4); 

BOOST_REQUIRE

REQUIRE型預言檢測

BOOST_REQUIRE(2+2==4); 

BOOST_WARN_MESSAGE

WARN型預言檢測,自定義日志

BOOST_WARN_MESSAGE2+2==4,"description…" );

BOOST_CHECK_MESSAGE

CHECK型預言檢測,自定義日志

BOOST_CHECK_MESSAGE2+2==4,"description…" );

BOOST_REQUIRE_MESSAGE

REQUIRE型預言檢測,自定義日志

BOOST_ REQUIRE _MESSAGE2+2==4,"description…" );

BOOST_ERROR

BOOST_CHECK_MESSAGE( false, M )

if( 2+2 !=4 )

BOOST_ERROR( "description…" );

BOOST_FAIL

BOOST_REQUIRE_MESSAGE( false, M )

if( 2+2 !=4 )

BOOST_FAIL( "description…" );

BOOST_WARN_EQUAL

WARN型左右值相等檢測檢測

BOOST_WARN_EQUAL(2+2,4);

BOOST_CHECK_EQUAL

CHECK型左右值相等檢測檢測

BOOST_CHECK_EQUAL(2+2,4);

BOOST_REQUIRE_EQUAL

REQUIRE型左右值相等檢測檢測

BOOST_REQUIRE_EQUAL(2+2,4);

該類還有不等,小于,大于,小于等于,大于等于的判別檢測,此處只羅列WARN的,其他不再一一羅列:BOOST_WARN_NE/ BOOST_WARN_LT/ BOOST_WARN_LE/ BOOST_WARN_GT/ BOOST_WARN_GE

BOOST_WARN_THROW

WARN型,判別執行函數期,有異常拋出。

BOOST_WARN_THROW(executeSql(“select…”),oracle::ErrorCodeException);

BOOST_CHECK_THROW

CHECK型,判別執行函數期,有異常拋出。

 

BOOST_REQUIRE_THROW

REQUIRE型,判別執行函數期,有異常拋出。

 

BOOST_WARN_EXCEPTION

WARN型,執行函數,當預言為真時,捕獲異常。

BOOST_WARN_THROW(executeSql(“select…”),oracle::ErrorCodeException,2+2==4);

 

BOOST_CHECK_EXCEPTION

CHECK型,執行函數,當預言為真時,捕獲異常。

 

BOOST_REQUIRE_EXCEPTION

REQUIRE型,執行函數,當預言為真時,捕獲異常。

 

BOOST_WARN_NO_THROW

WARN型,判別執行函數期,無異常拋出。

BOOST_WARN_NO_THROW(executeSql(“select…”));

BOOST_CHECK_NO_THROW

CHECK型,判別執行函數期,無異常拋出。

 

BOOST_REQUIRE_NO_THROW

REQUIRE型,判別執行函數期,無異常拋出。

 

BOOST_WARN_CLOSE

WARN型,判定左右值是否足夠逼近。用于浮點數比較。

BOOST_WARN_CLOSE2.1131,2.1132,0.01

BOOST_CHECK_CLOSE

CHECK型,判定左右值是否足夠逼近。用于浮點數比較

 

BOOST_REQUIRE_CLOSE

REQUIRE型,判定左右值是否足夠逼近。用于浮點數比較

 

BOOST_WARN_SMALL

WARN型,判定值是否足夠小(是否接近0)。用于浮點數比較

BOOST_WARN_CLOSE0.1,0.01

BOOST_CHECK_SMALL

CHECK型,判定值是否足夠?。ㄊ欠窠咏?/span>0)。用于浮點數比較

 

BOOST_REQUIRE_SMALL

REQUIRE型,判定值是否足夠?。ㄊ欠窠咏?/span>0)。用于浮點數比較

 

3.3 TestCase

對于TestCase/TestSuite等,Boost.Test既支持手動注冊方式,也支持自動注冊方式,當前Boost官方推薦自動注冊方式,手動注冊為了保持向前兼容保留,以后版本可能被移除。使用宏BOOST_AUTO_TEST_CASE即可自動注冊測試用例。使用如下:

#include <boost/test/unit_test.hpp>

BOOST_AUTO_TEST_CASE(test_case_name)

{

    BOOST_CHECK(true);

}

 

3.4 TestSuite

使用宏BOOST_AUTO_TEST_SUITE(test_suite_name)開始測試套件,使用BOOST_AUTO_TEST_SUITE_END()結束測試套件。使用舉例:

#include <boost/test/unit_test.hpp>

BOOST_AUTO_TEST_SUITE(test_suite_name)

BOOST_AUTO_TEST_CASE(test_case1)

{

    BOOST_CHECK(true);

}

BOOST_AUTO_TEST_CASE(test_case1)

{

    BOOST_CHECK(true);

}

BOOST_AUTO_TEST_SUITE_END()

3.5 MainTestSuite

使用宏BOOST_TEST_MODULE表明主測試套件,一個測試項目中只能存在一個主測試套件。使用舉例:

#define BOOST_TEST_MODULE maintest

#include <boost/test/unit_test.hpp>

3.6 TestFixture

測試夾具做測試前的準備工作和測試后的清理工作。而C++RAII機制(構造函數申請資源,析構函數釋放資源)恰好能滿足該需求。因此Boost中直接使用普通類做夾具。實現夾具舉例:該夾具在測試前將整數i初始化為5

struct MyFixture

{

    MyFixture():i(5){}

    ~MyFixture(){}

   Int I;

};

    夾具可以和TestCase一起使用,也可以和TestSuite一起使用,也可以和MainTestSuite一起使用。

    使用宏BOOST_FIXTURE_TEST_CASE(test_case_name, fixure_name)代替BOOST_AUTO_TEST_CASE(test_case_name)即可在TestCase中使用夾具,舉例如下:

BOOST_FIXTURE_TEST_CASE(test_case_name,MyFixture)

{

    BOOST_CHECK(i==5);

}

    使用宏BOOST_FIXTURE_TEST_SUITE(suite_name, fixure_name)代替BOOST_AUTO_TEST_SUITE(suite_name)即可在TestSuite中使用夾具,舉例如下:

BOOST_FIXTURE_TEST_SUITE(test_suite, MyFixture );

BOOST_AUTO_TEST_CASE (test_case1)

{

    BOOST_CHECK(i==5);

}

BOOST_AUTO_TEST_CASE(test_case2)

{

    BOOST_CHECK(++i==6);

}

BOOST_AUTO_TEST_SUITE_END()

       使用宏BOOST_GLOBAL_FIXTURE(MyFixture);MyFixture聲明為全局夾具,即可和MainTestSuite一起使用。

 

3.7 Boost.Test使用

對已經完成的項目做單元測試,假定該項目具有很好的測試性:

1) 對項目中的每個類對象創建一個測試套件,一個測試套件對應一個cpp文件。對類的每個類方法創建一個測試用例,這些測試用例均包含上前面的測試套件中。每個測試用例可以有多個測試斷言,對該方法進行充分測試。

2) 在測試主文件中定義宏BOOST_TEST_MODULE,并包含所有的測試套件文件。

3) Linux下,將被測試項目編譯成靜態庫(將main函數外的所有文件編譯打包)供測試項目連接。Window下為測試項目做靜態庫工程,設置測試工程依賴該工程。并將頭文件路徑設置正確,即可編譯運行。

附件為一示例項目以及對應單元測試工程舉例,項目目錄下make編譯生成靜態庫以及可執行程序,test目錄下make生成單元測試可執行程序。(略)
 

實行單元測試

4.1 現實困難

1、 內部依賴問題

類之間相互協作共同完成功能,類之間的依賴必不可少。為了測試某個類,必須實例化它依賴的類,而它依賴的類又可能依賴其他類,因此必須實例化其他類。如此一環扣一環,可能把整個項目大部分類都包含在了這次測試中,最后做的不是單元測試,而是掛著單元測試外殼的集成測試

2、 外部依賴問題

很多項目,尤其是我們的網絡應用服務器,運行期間需要依賴外部的其他服務器或者數據庫或者本地的文件系統。而對很多外部的依賴很難模擬,或者說模擬成本太高,往往讓測試者望而卻步。

3、 函數本身問題

項目中的很多或者可能是大部分函數,是沒有明確返回值或者無異常拋出,而只是和其他外設交互。難于使用測試斷言判定。

以上造成很難將某個類從項目中隔離出來,難以設置單元測試點。

4.2 解決困難

上述困難均為依賴造成。

1、內部解依賴

對被測代碼進行解依賴,強化設計,減少耦合,提高代碼可測性。解依賴的過程也即為對代碼重構過程,減少類間耦合,制造接口層。常用手段有:虛函數、函數指針、傳遞參數等方式。而對于難于進行解依賴情況,就要考慮提取分化重寫方法。

2、寫樁代碼模擬外部環境

單元測試不能直接依賴外部環境,必須寫樁代碼模擬。而外部環境的可模擬性與內部解依賴緊密相關。對于外部的隨機性和各種不確定性,樁代碼必須盡可能模擬。

3、開辟訪問類私有屬性通道

有些類方法雖然沒有明確返回值,但可能修改類的內部狀態,可以通過判斷類的私有屬性來判定類方法的執行情況??梢越o類增加Get()方法或者將私有屬性設置為protected。

更重要的是在代碼開發期,引入TDD思維,強化設計,提高代碼可測性,提高代碼的整體質量

posted on 2010-06-28 09:15 cppexplore 閱讀(4700) 評論(1)  編輯 收藏 引用

評論

# re: 【原創】單元測試分享 2010-06-28 16:17 陳梓瀚(vczh)
不管單元測試的框架支持多少功能,我們應該把項目寫成容易測試+容易維護+容易集成的形式,這也是單元測試可以改進項目代碼的一個地方。如果單元測試可以成為一個指標的話,那么項目代碼就會寫得更好一點了。  回復  更多評論
  

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            亚洲国产精品久久久久久女王| 亚洲欧美日韩一区二区| 久久综合色8888| 久久久久久久久久久成人| 在线播放日韩欧美| 久久久久久综合| 欧美99在线视频观看| 亚洲经典三级| 99精品热视频| 国产乱码精品一区二区三区五月婷| 欧美在线观看日本一区| 久久人人精品| 亚洲一区www| 欧美在线在线| 一区二区电影免费在线观看| 亚洲欧美文学| 亚洲人体偷拍| 欧美一区二区观看视频| 亚洲精品偷拍| 午夜精品久久久久久久蜜桃app| 精品成人久久| 夜夜躁日日躁狠狠久久88av| 国产综合av| 99视频精品免费观看| 国外成人在线| 亚洲一区国产一区| 亚洲精品少妇网址| 亚洲无吗在线| 日韩视频在线永久播放| 久久国产免费看| 亚洲欧美日韩一区二区在线| 久久综合中文| 久久国产主播| 国产精品www网站| 欧美激情第五页| 国内精品久久久久伊人av| 一区二区三区欧美在线| 亚洲国产成人午夜在线一区| 午夜精品在线观看| 一个色综合导航| 免费在线看成人av| 久久久久久亚洲精品中文字幕| 欧美日韩亚洲不卡| 最新69国产成人精品视频免费| 极品少妇一区二区三区| 亚洲与欧洲av电影| 亚洲一区二区不卡免费| 欧美精品成人在线| 欧美二区不卡| 亚洲国产欧美精品| 久久久欧美一区二区| 久久久久久一区二区| 国产精品网站在线观看| 亚洲图片欧洲图片av| 中文亚洲免费| 国产精品v日韩精品| 一区二区三区精密机械公司| 在线视频免费在线观看一区二区| 欧美噜噜久久久xxx| 亚洲精品婷婷| 亚洲图片你懂的| 国产精品第一页第二页第三页| 亚洲精品美女在线观看播放| 99精品免费网| 欧美午夜精品久久久久久久| 99热免费精品| 欧美一区二区成人6969| 国产午夜精品理论片a级探花| 新狼窝色av性久久久久久| 久久激情综合| 怡红院精品视频| 蜜臀av一级做a爰片久久| 久久久久久午夜| 亚洲福利小视频| 久久久中精品2020中文| 欧美v国产在线一区二区三区| 亚洲东热激情| 欧美日韩一二区| 午夜视频一区在线观看| 老司机午夜精品视频在线观看| 伊人精品成人久久综合软件| 欧美成人精品激情在线观看| 亚洲日本免费电影| 亚洲一区在线视频| 国内揄拍国内精品久久| 欧美成人福利视频| 这里只有精品视频| 久久婷婷久久| 国产精品99久久久久久人| 国产伦精品一区二区三区免费 | 小黄鸭精品aⅴ导航网站入口| 蜜桃伊人久久| 在线亚洲欧美视频| 国产视频一区二区三区在线观看| 久久深夜福利免费观看| 日韩视频不卡| 美女主播精品视频一二三四| 99国产精品久久久久久久久久| 国产乱码精品一区二区三区不卡| 久久在线观看视频| 亚洲自拍三区| 亚洲国产精品va在线观看黑人| 午夜精品福利一区二区蜜股av| 一区二区三区在线免费播放| 欧美性猛片xxxx免费看久爱| 久久免费99精品久久久久久| 99精品国产99久久久久久福利| 久久中文字幕一区| 亚洲中无吗在线| 亚洲全部视频| 狠狠色伊人亚洲综合网站色| 国产精品第13页| 欧美日韩高清一区| 久久综合99re88久久爱| 欧美亚洲尤物久久| 99亚洲精品| 亚洲国产精品国自产拍av秋霞 | 亚洲欧美文学| 日韩一级不卡| 亚洲国产影院| 黑丝一区二区三区| 国产日韩欧美黄色| 国产精品久久久久av免费| 欧美伦理一区二区| 免费在线亚洲欧美| 久久综合伊人77777| 欧美在线国产| 欧美亚洲免费高清在线观看| 中国成人黄色视屏| 中文久久乱码一区二区| 日韩一级欧洲| 日韩视频免费| 亚洲人成在线观看一区二区| 亚洲电影免费| 亚洲成在人线av| 亚洲高清免费视频| 亚洲电影免费观看高清完整版在线| 免费人成网站在线观看欧美高清| 久久综合狠狠综合久久激情| 久久精品亚洲一区二区| 久久国产精品99国产精| 久久国产一区| 国产精品久久一区主播| 欧美一级久久久| 欧美亚洲一区在线| 欧美一级午夜免费电影| 欧美一区二区精品在线| 久久国产色av| 免播放器亚洲一区| 欧美久久久久| 国产精品久久久久永久免费观看| 国产精品久久久久久久第一福利| 国产精品日韩一区二区三区| 国产精品―色哟哟| 狠狠色丁香婷婷综合久久片| 精品999日本| 日韩视频免费观看高清在线视频| 亚洲色图综合久久| 欧美在线观看视频在线| 久久全国免费视频| 亚洲国产天堂久久综合| 亚洲午夜av电影| 欧美中文字幕在线| 欧美国产日韩在线观看| 欧美性一区二区| 激情综合中文娱乐网| 99精品免费| 久久久久网址| 亚洲精品在线免费观看视频| 亚洲永久免费精品| 老司机午夜精品视频在线观看| 欧美日韩亚洲成人| 国产综合久久| 99在线精品视频| 久久久久久网| 一区二区三区免费看| 久久亚洲综合| 国产麻豆9l精品三级站| 亚洲精品四区| 久久精品一区二区三区不卡牛牛 | 亚洲综合国产| 久久影视三级福利片| 99成人免费视频| 久久久久久久高潮| 国产精品久久久一区二区| 亚洲国产综合视频在线观看| 性视频1819p久久| 亚洲人成网站影音先锋播放| 久久久精品国产免费观看同学| 欧美日韩在线直播| 亚洲精品国产精品乱码不99| 久久精品日韩| 亚洲一区高清| 欧美日韩综合久久| 亚洲精品久久久久久下一站| 久久福利影视| 香蕉尹人综合在线观看| 国产精品日韩欧美综合 | 日韩午夜中文字幕| 蜜臀久久久99精品久久久久久|