




版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請(qǐng)進(jìn)行舉報(bào)或認(rèn)領(lǐng)
文檔簡(jiǎn)介
第第頁(yè)開發(fā)和測(cè)試工程師寫代碼的高下之分開發(fā)和測(cè)試工程師寫代碼的高下之分
發(fā)表于:2023-04-18來源::點(diǎn)擊數(shù):標(biāo)簽:代碼測(cè)試工程師開發(fā)高下
有來javascript:;"onclick="javascript:tagshow(event,'%C3%E6%CA%D4');"target="_self"_base_target="_blank">面試的同行問個(gè)一個(gè)問題:“如果微軟的開發(fā)和測(cè)試工程師都需要寫代碼,那么兩者寫出來的代碼有高下之分嗎?”當(dāng)時(shí)我只能簡(jiǎn)單的解釋
有來javascript:;"onclick="javascript:tagshow(event,'%C3%E6%CA%D4');"target="_self"_base_target="_blank">(面試)的同行問個(gè)一個(gè)問題:“如果(微軟)的開發(fā)和(測(cè)試)工程師都需要寫代碼,那么兩者寫出來的代碼有高下之分嗎?”
當(dāng)時(shí)我只能簡(jiǎn)單的解釋一下?,F(xiàn)在可以多說一些了。
舉個(gè)例子,(單元測(cè)試)。適合不同語言的工具有一大堆,各個(gè)論壇上都能搜到大堆文章。出現(xiàn)頻率最高的不外乎CppUnit,NUnit,JUnit三種。比起NUnit和JUnit,CppUnit因?yàn)镃++語言特性的關(guān)系,用起來較不方便。
這里我給大家秀一下,解決這個(gè)問題,測(cè)試工程師會(huì)如何的“不擇手段”。
單元測(cè)試往往需要解決以下幾個(gè)問題:
1.用戶能在產(chǎn)品代碼中指定需要測(cè)試的函數(shù)
2.用戶能在測(cè)試代碼中指定需要執(zhí)行的函數(shù)
3.用戶能指定各種控制執(zhí)行過程的參數(shù),比如優(yōu)先級(jí)、重復(fù)、初始化/清理函數(shù)等等
(其它)就先不說了,待會(huì)大家就知道再多的都能做到,現(xiàn)在先做到這三個(gè)需求就挺不錯(cuò)了。需求1是可選的。VisualStudio從2023開始有這個(gè)功能,但是如果沒有,估計(jì)大家也不會(huì)太在意,對(duì)吧?
那么,CppUnit如果要做到NUnit和JUnit的樣子會(huì)遇到什么困難呢?首先就是C++缺乏反射(Reflection)功能。
你看NUnit和JUnit都是定義了一大堆a(bǔ)ttribute。用戶通過為函數(shù)指定恰當(dāng)?shù)腶ttribute,就能標(biāo)明這個(gè)函數(shù)需要作為(test)case執(zhí)行,之前執(zhí)行初始化函數(shù)Setup,之后執(zhí)行Cleanup,還有重復(fù)100次,優(yōu)先級(jí)2什么的,其它的往上堆a(bǔ)ttribute就行了。
VisualStudio2023之后為實(shí)現(xiàn)需求1,通過.NETReflection找出一個(gè)類的全部成員函數(shù),然后列個(gè)表讓你選要測(cè)試的函數(shù),最后代碼架子都給你搭好。
而CppUnit得用模板,還有另外一個(gè)我忘了名字的C++單元測(cè)試工具用的是宏。他們費(fèi)這么些勁是為了什么呢?其實(shí)是為了取得這些信息:
具有特定標(biāo)記的函數(shù)的名字或者入口地址/函數(shù)指針
附加在該函數(shù)上的各種整數(shù),字符串或者函數(shù)指針的值
拿到之后呢?CppUnit也好,NUnit或JUnit也好,都是按照取得的信息規(guī)劃每個(gè)函數(shù)的執(zhí)行,保證異常和錯(cuò)誤不干擾其它函數(shù)的執(zhí)行,統(tǒng)計(jì)整理執(zhí)行結(jié)果和記錄(日志),沒什么區(qū)別。
我們來看看,Reflection對(duì)于取得那些信息是不是必需的呢?
沒錯(cuò),Reflection是能夠在執(zhí)行時(shí)(runtime)取得上述信息,然后在執(zhí)行時(shí)利用它指導(dǎo)testcase的執(zhí)行。
等等,我雞蛋里挑個(gè)骨頭:非要把“取得信息”和“指導(dǎo)testcase執(zhí)行”放在同一段執(zhí)行時(shí)期里面嗎?或者說,先“取得信息”,停一下,再“指導(dǎo)testcase執(zhí)行”,行不行?
甚至,“取得信息”放在一個(gè)程序里先執(zhí)行,“指導(dǎo)testcase執(zhí)行”放在另一個(gè)程序里后執(zhí)行,又如何?
更甚之,“取得信息”是讓別的程序做的,之后讓“指導(dǎo)testcase執(zhí)行”撿現(xiàn)成的呢?
可見,Reflection提供了比我們所需要的多得多的功能,實(shí)際上我們只需要知道“怎樣指導(dǎo)testcase執(zhí)行”就夠了。
那可以如何實(shí)現(xiàn)呢?
一、編譯鏈接兩次
對(duì),你沒聽錯(cuò)!編譯鏈接之后除了可執(zhí)行文件還會(huì)產(chǎn)生不少有用信息,最有用的當(dāng)屬PDB文件,包含了所有標(biāo)識(shí)符(Symbol)的信息。VisualStudio2023開始就帶有一個(gè)DIASDK,可以在其安裝目錄的DIASDK文件夾下面找到頭文件、庫(kù)文件、COM組件DLL和示例程序。用它分析一個(gè)DLL或者EXE可執(zhí)行文件對(duì)應(yīng)的PDB文件,你可以取得每個(gè)編譯單元(用.c/.cpp編譯得到的obj或者lib)里全部函數(shù)的情況,包括函數(shù)名字、是否成員函數(shù)、返回類型、全部參數(shù)的名字和類型、全部局部變量的名字和類型,甚至它在哪個(gè)文件的哪一行定義。
在這些信息中可以過濾出用戶預(yù)先指定的信息,用來拼成另一個(gè)C/C++源文件,這個(gè)源文件叫做執(zhí)行表,里面包含了所有需要執(zhí)行函數(shù)的名字列表以及各項(xiàng)參數(shù)的靜態(tài)定義?!爸笇?dǎo)testcase執(zhí)行”是可以預(yù)先分離實(shí)現(xiàn)的模塊,把它include進(jìn)來即可。最后,把原先用來產(chǎn)生可執(zhí)行文件的全部文件,把定義main或者DllMain的那個(gè)源文件,改為執(zhí)行表,再編譯鏈接一次,大功告成。
問題是怎樣產(chǎn)生第一個(gè)可執(zhí)行文件呢?用戶使用單元測(cè)試工具的時(shí)候不都實(shí)現(xiàn)了一些函數(shù)嗎?那就能產(chǎn)生用戶的編譯單元。我們可以預(yù)先提供一個(gè)定義DllMain的殼lib,它與用戶的編譯單元鏈接在一起就成為被DIASDK分析的DLL。然后像前面說的,最后換成執(zhí)行表的編譯單元。
你注意到了嗎,用這個(gè)方法,別說CppUnit,做CUnit都可以。
二、用struct表達(dá)attributes
剛才并沒有提到如何像NUnit和JUnit一樣取得attribute所定義的信息。我們要求需要執(zhí)行的函數(shù)定義成返回BOOL類型,只有一個(gè)參數(shù),一個(gè)結(jié)構(gòu)的指針。
如果用戶定義的這個(gè)結(jié)構(gòu)像這樣
structSIGNATURE_001
{
SIGNATURE_001()
{
constchar*title="Testcase1";
constDWORDpriority=PRIORITY.P1;
constchar*Setup="MySetup";
constchar*Cleanup="MyCleanup";
…
}
};
用DIASDK分析就會(huì)發(fā)現(xiàn),這個(gè)函數(shù)的參數(shù)類型具有構(gòu)造函數(shù),其中有局部的常量名字如何,值多少。依靠這些信息,我們足可以判斷一個(gè)函數(shù)是不是需要執(zhí)行,所需要的參數(shù)都是怎么樣。
為了簡(jiǎn)化,我們可以用一個(gè)簡(jiǎn)單的宏幫助定義這一切,放在函數(shù)前面就可以了。當(dāng)然它不是必需的。
在C下面可不能這么用,但辦法還是有的。
三、用解釋表達(dá)attributes
既然我們能把函數(shù)定位到某一行,那么往前掃描源文件行,遇到三個(gè)斜杠開頭的就濾掉“///”讀進(jìn)來,如果我們預(yù)期這些解釋看起來是這個(gè)樣子
///TestCase
///TitleTestcase1/Title
///Priority1/Priority
///SetupMySetup/Setup
///CleanupMyCleanup/Cleanup
////TestCase
那么只要分析這段XML文本,接下來的事情就跟前面說的一樣了。
四、自動(dòng)運(yùn)行整個(gè)過程
用戶需要的是寫好測(cè)試代碼,執(zhí)行一個(gè)命令就能得到所有可執(zhí)行代碼。我們可以利用makefile把這一切連接起來:先寫好需要分析的DLL的依賴關(guān)系,然后讓執(zhí)行表依賴上述DLL,命令為執(zhí)行分析代碼產(chǎn)生執(zhí)行表,最后讓目標(biāo)DLL依賴執(zhí)行表即可。又或者用VisualStudio的BuildSteps來驅(qū)動(dòng)也可以。
到這里,大家可以看到,為了實(shí)現(xiàn)最終目的,我們突破了習(xí)慣的限制(只使用語言特性),并且充分利用現(xiàn)有的(技術(shù))和工具(DIASDK和XMLParser)。只要能實(shí)現(xiàn)目的,“無所不用其極”。
你可能覺得折騰這么一套東西動(dòng)作也挺大的。我得說,“看菜吃飯”。
另一個(gè)例子,有一個(gè)測(cè)試框架,萬事俱備,就是沒法把testcase自動(dòng)傳送到AppleMacintosh的機(jī)器上?,F(xiàn)有的代碼可以讓testcase在AppleMacintosh上執(zhí)行,也可以把testcase從服務(wù)器下載到Windows');"target="_self"_base_target="_blank">(Windows)測(cè)試機(jī)器上發(fā)動(dòng)執(zhí)行,但是沒法跟AppleMacintosh交流。
怎么辦?在Apple上開發(fā)誰都不懂。在AppleMacintosh上寫一個(gè)客戶端跟服務(wù)器交流,夠忙半天的了。面對(duì)一整套已經(jīng)完備的測(cè)試框架,讓它盡快用于新的環(huán)境,比做什么都重要。
別人告訴我,可以AppleMacintosh上開一個(gè)共享夾,然后Windows的機(jī)器可以用UNC路徑往里面讀寫文件。
OK,這就足夠了。Windows測(cè)試機(jī)器上發(fā)動(dòng)執(zhí)行的只是一個(gè)腳本,把需要用到的文件往指定Apple機(jī)器的共享文件夾上寫。寫完之后再寫一個(gè)文件,名字是約定好的,例如“ready”,里面包含啟動(dòng)testcase的命令行。然后不停的隔一段時(shí)間檢查共享文件夾里面一個(gè)叫做例如“done”的文件,出現(xiàn)之后把它作為結(jié)果返回服務(wù)器,最后把它和其它文件都刪掉,退出。
AppleMacintosh上面則運(yùn)行另一個(gè)腳本,始終不退出。它不停的隔一段時(shí)間檢查其指定的共享文件夾里面一個(gè)叫做“ready”的文件,出現(xiàn)之后執(zhí)行里面的命令并且等待它結(jié)束。這個(gè)命令必須生成一個(gè)叫做“done”的文件,包含執(zhí)行結(jié)果。然后,不停的隔一段時(shí)間檢查“done”是不是還在,不在了就回到最初的檢查“ready”的代碼。
這就足夠了。兩個(gè)腳本加起來50行不到。
你覺得它太粗糙了吧?這么簡(jiǎn)單的協(xié)議?
事實(shí)上,它并不需要十分健壯。
一、Windows和Macintosh雙方的網(wǎng)絡(luò)文件系統(tǒng)協(xié)議解決了很多問題
二、測(cè)試機(jī)器是不會(huì)有人去用的,你可以安全的假設(shè)只有你的程序在執(zhí)行
三、服務(wù)器和testcase都已經(jīng)測(cè)試過,他們應(yīng)該負(fù)擔(dān)起若干健壯性的需求。事實(shí)上,他們比這兩個(gè)腳本更適合做這個(gè),不是嗎?
這就是“看菜吃飯”:不需要的功能,是不需要去實(shí)現(xiàn)的,無論它看上去有多么的cool;必需的功能,無論如何都要做到,無論它看上去有多么的boring。
其實(shí),無論開發(fā)測(cè)試,都是為了讓人們更好的發(fā)揮自身的潛力。開發(fā)工程師讓人們可以專注于自身的事業(yè)而不用過多(
溫馨提示
- 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請(qǐng)下載最新的WinRAR軟件解壓。
- 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請(qǐng)聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
- 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁(yè)內(nèi)容里面會(huì)有圖紙預(yù)覽,若沒有圖紙預(yù)覽就沒有圖紙。
- 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
- 5. 人人文庫(kù)網(wǎng)僅提供信息存儲(chǔ)空間,僅對(duì)用戶上傳內(nèi)容的表現(xiàn)方式做保護(hù)處理,對(duì)用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對(duì)任何下載內(nèi)容負(fù)責(zé)。
- 6. 下載文件中如有侵權(quán)或不適當(dāng)內(nèi)容,請(qǐng)與我們聯(lián)系,我們立即糾正。
- 7. 本站不保證下載資源的準(zhǔn)確性、安全性和完整性, 同時(shí)也不承擔(dān)用戶因使用這些下載資源對(duì)自己和他人造成任何形式的傷害或損失。
最新文檔
- 二零二五年度水上樂園游泳館場(chǎng)地租賃與水上樂園配套設(shè)施租賃協(xié)議
- 2025年度老舊小區(qū)外墻改造工程安全責(zé)任合同
- 二零二五年度國(guó)際貿(mào)易信用證業(yè)務(wù)代理及風(fēng)險(xiǎn)管理協(xié)議
- 海洋漁業(yè)資源保護(hù)與海產(chǎn)品銷售一體化合同
- 二零二五年度企業(yè)用工協(xié)議與勞動(dòng)權(quán)益保障與員工激勵(lì)機(jī)制合同
- 二零二五年度廠房裝修施工安全責(zé)任與綠色施工標(biāo)準(zhǔn)協(xié)議書
- 2025年度酒店與旅游紀(jì)念品店合作經(jīng)營(yíng)合同
- 二零二五年度籃球活動(dòng)參與者免責(zé)責(zé)任協(xié)議
- 二零二五年度汽車美容店員工勞動(dòng)爭(zhēng)議解決合同模板
- 二零二五年度農(nóng)村房屋贈(zèng)與合同附農(nóng)業(yè)保險(xiǎn)合作協(xié)議
- 經(jīng)濟(jì)學(xué)彈性理論課件
- 公司員工獎(jiǎng)懲制度流程
- 星巴克案例分析-星巴克成功之道
- 把未來點(diǎn)亮歌詞打印版
- 危險(xiǎn)化學(xué)品建設(shè)項(xiàng)目竣工驗(yàn)收?qǐng)?bào)告
- 國(guó)家中醫(yī)藥管理局第3批24個(gè)專業(yè)104個(gè)病種中醫(yī)診療方案
- 婦產(chǎn)科學(xué)(第9版)第三章 女性生殖系統(tǒng)生理
- LY/T 2241-2014森林生態(tài)系統(tǒng)生物多樣性監(jiān)測(cè)與評(píng)估規(guī)范
- GB/T 9086-2007用于色度和光度測(cè)量的標(biāo)準(zhǔn)白板
- 2023年山東力明科技職業(yè)學(xué)院高職單招(數(shù)學(xué))試題庫(kù)含答案解析
- GB/T 24338.4-2018軌道交通電磁兼容第3-2部分:機(jī)車車輛設(shè)備
評(píng)論
0/150
提交評(píng)論