• <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>

            CppExplore

            一切像霧像雨又像風(fēng)

              C++博客 :: 首頁(yè) :: 新隨筆 :: 聯(lián)系 :: 聚合  :: 管理 ::
              29 隨筆 :: 0 文章 :: 280 評(píng)論 :: 0 Trackbacks
            作者:CppExplore 網(wǎng)址:http://www.shnenglu.com/CppExplore/

            在c++的世界里,程序設(shè)計(jì)的優(yōu)雅讓位于程序的穩(wěn)定性、健壯性。“好程序是測(cè)出來(lái)的”這句話在C++領(lǐng)域里得到了充分體現(xiàn)。下面是我在開(kāi)發(fā)中使用的測(cè)試方法,拋磚引玉,和大家交流下。
            測(cè)試期間,關(guān)閉對(duì)core文件的限制,使用命令:ulimit -c unlimited
            (1)開(kāi)發(fā)階段,使用cppunit維護(hù)測(cè)試用例。我一般是用于測(cè)試解析類、算法類。
            從http://sourceforge.net/projects/cppunit/下載最新版本,解壓,看安裝文檔,一般是./configure & make & make install。
            下面舉例說(shuō)明我使用cppunit的方法。假設(shè)自己的源碼位于src目錄下,里面有class1.h/class1.cpp/class2.h/class2.cpp。相對(duì)src建立平級(jí)目錄test存放測(cè)試工程,為class1/class2分別建立測(cè)試類文件testClass1.h/testClass2.h,建立main函數(shù)所在文件test.cpp、makefile。
            testClass1.h代碼如下,testClass2.h類似。

            #include "class1.h"
            #include <iostream>
            #include "cppunit/TestRunner.h"
            #include "cppunit/TestResult.h"
            #include "cppunit/TestResultCollector.h"
            #include "cppunit/extensions/HelperMacros.h"
            #include "cppunit/BriefTestProgressListener.h"
            #include "cppunit/extensions/TestFactoryRegistry.h"
            #include "cppunit/TextOutputter.h"
            #include "cppunit/CompilerOutputter.h"
            #include "cppunit/TestCaller.h"
            class testClass1:public CPPUNIT_NS::TestFixture
            {
               CPPUNIT_TEST_SUITE(testClass1);
               CPPUNIT_TEST(testCase1);
               CPPUNIT_TEST(testCase2);
               CPPUNIT_TEST_SUITE_END();
                public:
                    virtual void setUp(){}
                    virtual void tearDown(){}
                    void testCase1()
                {
                        testClass1 a;
                        a.oper..;
                        CPPUNIT_ASSERT_EQAL(a.get..,);
                    }
                    void testCase2()
                {
                        CPPUNIT_ASSERT(==);
                    }
            };

            test.cpp代碼如下:
            #include "testClass1.h"
            #include "testClass2.h"
            #include <iostream>
            #include "cppunit/TestRunner.h"
            #include "cppunit/TestResult.h"
            #include "cppunit/TestResultCollector.h"
            #include "cppunit/extensions/HelperMacros.h"
            #include "cppunit/BriefTestProgressListener.h"
            #include "cppunit/extensions/TestFactoryRegistry.h"
            #include "cppunit/TextOutputter.h"
            #include "cppunit/CompilerOutputter.h"
            #include "cppunit/TestCaller.h"
             CPPUNIT_TEST_SUITE_REGISTRATION(testClass1);
             CPPUNIT_TEST_SUITE_REGISTRATION(testClass1);
            int main()
              {
                CPPUNIT_NS::TestResult controller;
                CPPUNIT_NS::TestResultCollector result;
                controller.addListener( &result );        
                CPPUNIT_NS::TestRunner runner;
                runner.addTest( CPPUNIT_NS::TestFactoryRegistry::getRegistry().makeTest() );
                runner.run( controller );
                CPPUNIT_NS::CompilerOutputter out( &result, std::cout );
                out.write();
                return 0;
            }

            makefile文件如下:
            EXE=test
            SRC=test.cpp
            INC_PATH=-I ../src -I (cppunit頭文件的目錄) -I(依賴的其他頭文件路徑)
            LIB_PATH=-L (cppunit動(dòng)態(tài)庫(kù)所在的目錄) -L (依賴的其他庫(kù)所在目錄)
                LIB=-lcppunit -ldl 
            all:
                g++ $(SRC) $(LIB_PATH) $(LIB)  $(INC_PATH) -o $(EXE)

            再有新的需要測(cè)試類,增加相應(yīng)的測(cè)試類,稍微修改下test.cpp即可(增加一句#include,一句CPPUNIT_TEST_SUITE_REGISTRATION)。
            保證開(kāi)發(fā)結(jié)束后,解析類、算法類等不會(huì)有錯(cuò)誤。
            (2)白盒測(cè)試階段。
            這個(gè)基本是功能邏輯性測(cè)試,檢測(cè)所有數(shù)據(jù)結(jié)構(gòu)按要求變化以及保證各線程之間變化的一致性。這是最基本也是最全面的一次測(cè)試,保證測(cè)試的功能覆蓋率100%。白盒測(cè)試期間可以在代碼里加一些宏編譯選項(xiàng)或者增加程序交互功能用于觀察所有數(shù)據(jù)結(jié)構(gòu)的變化。
            保證測(cè)試完畢沒(méi)有功能性、邏輯性的錯(cuò)誤。
            (3)內(nèi)存測(cè)試階段。使用valgrind檢測(cè)顯式內(nèi)存泄漏、內(nèi)存讀寫錯(cuò)誤。
            從http://www.valgrind.org/下載最新版本,解壓,看安裝文檔,一般是./configure & make & make install。
            檢測(cè)內(nèi)存一般使用命令valgrind --tool=memcheck -v --leak-check=full ./待測(cè)程序錯(cuò)誤的地方會(huì)用==×××==(×××表示數(shù)字)標(biāo)出。
            使用一路模擬客戶端做陪測(cè)。
            保證測(cè)試完畢,單路客戶端陪測(cè)的情況下沒(méi)有任何的顯式內(nèi)存泄漏,沒(méi)有任何的內(nèi)存讀寫錯(cuò)誤。
            (4)寫批量客戶端模擬程序。建議熟悉一門方便socket編程的腳本語(yǔ)言,推薦perl。腳本語(yǔ)言簡(jiǎn)單,實(shí)現(xiàn)快速,特適合做陪測(cè)。
            首先寫一個(gè)能讀取配置文件信息,按配置文件的要求向相應(yīng)的server,按配置文件的流程發(fā)送信令的perl程序。
            下面是我rtsp相關(guān)的一個(gè)server陪測(cè)的配置文件:

            ip=127.0.0.1
            port=9115
            url=rtsp://172.24.202.190:554/asset/service?USERID=320101312345670001&ChanelNo-PUID=0-320101000200000001&PlayMethod=0
            <s,2>
            <p,2>
            <u,2>
            <p,2>
            <t,2>

            其中ip是server IP,port是rtsp端口,url是發(fā)送信令帶的url。<>表示按順序發(fā)送的信令,這個(gè)配置文件表示先發(fā)送一個(gè)setup,然后sleep 2秒,再發(fā)送一個(gè)play,然后sleep 2秒,繼續(xù)......這個(gè)程序可作為(3)中的陪測(cè)程序

            在上面程序的基礎(chǔ)上修改,讀取配置文件后,死循環(huán)按順序發(fā)送信令,假設(shè)該程序稱做B。
            寫一個(gè)新的perl文件,完成如下功能,起幾十路使用某配置文件的B程序,sleep幾秒后,再起幾十路使用其它配置文件的B程序.....,或者一起起也可以,自行設(shè)計(jì),最后killall所有,從頭循環(huán)運(yùn)行。
            總之盡可能的模擬客戶端的所有行為,包括突然斷聯(lián)等,并且保證一定的壓力。
            (5)壓力下測(cè)試內(nèi)存。繼續(xù)在valgrind下測(cè)試,使用(4)中的測(cè)試腳本做配測(cè)。
            保證壓力下無(wú)異常狀態(tài)、無(wú)數(shù)據(jù)不一致?tīng)顟B(tài)、無(wú)顯式內(nèi)存泄漏、無(wú)內(nèi)存讀寫異常
            (6)穩(wěn)定性以及內(nèi)存泄漏測(cè)試。
            陪測(cè)腳本起幾百路客戶端,保證主程序的cpu占用率在70%以上,持續(xù)運(yùn)行20多小時(shí)。
            測(cè)試期間,關(guān)注進(jìn)程對(duì)內(nèi)存的占用率,是保持在恒定水平還是隨運(yùn)行時(shí)間的增長(zhǎng)而增長(zhǎng)。
            測(cè)試完畢,保證主程序負(fù)荷運(yùn)行長(zhǎng)時(shí)間而不宕機(jī)、沒(méi)有內(nèi)存泄漏發(fā)生。
            (7)代碼覆蓋率測(cè)試。gcov
            gcov是隨gcc安裝的,可以檢查陪測(cè)程序?qū)δ繕?biāo)程序的代碼覆蓋情況。
            不斷修改測(cè)試腳本,保證測(cè)試盡量全面。代碼被執(zhí)行的次數(shù)也可以做為以后性能測(cè)試的一個(gè)參考。
            (8)性能測(cè)試。gprof
            同gcov一樣,gprof也是隨gcc安裝的,它可以檢測(cè)目標(biāo)程序中所有函數(shù)的調(diào)用時(shí)間,并根據(jù)消耗時(shí)間排序,方便找出性能瓶頸。
            找出系統(tǒng)的主要性能瓶頸,經(jīng)過(guò)性能測(cè)試后,一般會(huì)發(fā)現(xiàn)影響系統(tǒng)的主要因素還是數(shù)據(jù)結(jié)構(gòu)和算法。

            測(cè)試期間,任何的coredump/任何的內(nèi)存讀寫異常,務(wù)必處理掉。墨菲法則說(shuō),一個(gè)事情如果有可能變?cè)悖聦?shí)則是會(huì)變的更糟。任何一個(gè)微小的、出現(xiàn)幾率極小的bug,如果不在研發(fā)測(cè)試階段解決,都可能造成以后更大代價(jià)的返工,甚至給客戶的運(yùn)營(yíng)帶來(lái)災(zāi)難。希望在每個(gè)人身上生效的都是馬太效應(yīng),而不是墨菲法則。

            以上都是我個(gè)人摸索的結(jié)果,沒(méi)有參與過(guò)測(cè)試培訓(xùn),也沒(méi)有和其他同事交流過(guò),因此可能有閉門造車的嫌疑,還請(qǐng)看這篇文章的高手們不吝賜教。

            posted on 2007-11-15 19:09 cppexplore 閱讀(3677) 評(píng)論(6)  編輯 收藏 引用

            評(píng)論

            # re: 【原創(chuàng)】c++ server測(cè)試全攻略[未登錄](méi) 2007-11-15 23:15 thinkry
            我頂  回復(fù)  更多評(píng)論
              

            # re: 【原創(chuàng)】c++ server測(cè)試全攻略 2007-11-16 09:26 金慶
            至少有借鑒意義。內(nèi)存泄漏有顯式與隱式之分嗎,請(qǐng)教?  回復(fù)  更多評(píng)論
              

            # re: 【原創(chuàng)】c++ server測(cè)試全攻略[未登錄](méi) 2007-11-16 11:39 cppexplore
            @金慶
            呵呵 不好意思 沒(méi)說(shuō)明白
            這里的顯式是說(shuō)能被valgrind直接測(cè)試出來(lái)了

            最終的內(nèi)存泄漏還是要看穩(wěn)定性測(cè)試的時(shí)候 占用的內(nèi)存百分比不隨時(shí)間的增加而增長(zhǎng)  回復(fù)  更多評(píng)論
              

            # re: 【原創(chuàng)】c++ server測(cè)試全攻略 2007-11-16 17:39 海邊沫沫
            收藏  回復(fù)  更多評(píng)論
              

            # re: 【原創(chuàng)】c++ server測(cè)試全攻略 2007-12-16 14:33 秦歌
            頂!以后多交流,http://www.shnenglu.com/xczhang  回復(fù)  更多評(píng)論
              

            # re: 【原創(chuàng)】測(cè)試系列之 c++ server測(cè)試全攻略 2008-06-18 22:08 zichuanxiu
            可以考慮 一些異常測(cè)試,資源受限,比如內(nèi)存分配失敗測(cè)試,定時(shí)器啟動(dòng)失敗測(cè)試等。  回復(fù)  更多評(píng)論
              


            只有注冊(cè)用戶登錄后才能發(fā)表評(píng)論。
            網(wǎng)站導(dǎo)航: 博客園   IT新聞   BlogJava   博問(wèn)   Chat2DB   管理


            国产69精品久久久久9999| 久久久精品国产sm调教网站 | 国产精品九九久久免费视频| 久久青青草原精品影院| 久久精品国产黑森林| 久久久精品人妻一区二区三区蜜桃| 久久婷婷成人综合色综合| 国产成人精品久久亚洲| 久久久www免费人成精品| 91久久精品国产成人久久| 尹人香蕉久久99天天拍| 久久人人爽人人爽人人片av高请| 伊人久久免费视频| 精品久久久久久中文字幕大豆网| av无码久久久久久不卡网站| 久久精品国产欧美日韩99热| 97久久天天综合色天天综合色hd| 欧美日韩中文字幕久久久不卡| 久久久久久九九99精品| 模特私拍国产精品久久| 99热精品久久只有精品| 99久久99久久精品免费看蜜桃 | 精品久久久久久国产牛牛app| 伊人久久大香线焦AV综合影院| 国产精品热久久无码av| 嫩草影院久久99| 国内精品伊人久久久久AV影院| 久久亚洲AV无码精品色午夜| 色综合合久久天天给综看| 丁香久久婷婷国产午夜视频| 国产一区二区精品久久| 国产亚洲精久久久久久无码| 久久亚洲日韩精品一区二区三区| 久久久久久曰本AV免费免费| 久久综合日本熟妇| 一级做a爰片久久毛片免费陪| 久久精品一区二区三区中文字幕 | 成人精品一区二区久久久| 久久香蕉国产线看观看乱码| aaa级精品久久久国产片| 久久精品国产91久久麻豆自制|