版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請進行舉報或認領(lǐng)
文檔簡介
第12章面向?qū)ο髮崿F(xiàn)12.1程序設(shè)計語言12.2程序設(shè)計風格12.3測試策略12.4設(shè)計測試用例12.5小結(jié)習(xí)題面向?qū)ο髮崿F(xiàn)主要包括兩項工作:把面向?qū)ο笤O(shè)計結(jié)果翻譯成用某種程序語言書寫的面向?qū)ο蟪绦?;測試并調(diào)試面向?qū)ο蟮某绦颉C嫦驅(qū)ο蟪绦虻馁|(zhì)量基本上由面向?qū)ο笤O(shè)計的質(zhì)量決定,但是,所采用的程序語言的特點和程序設(shè)計風格也將對程序的可靠性、可重用性及可維護性產(chǎn)生深遠影響。目前,軟件測試仍然是保證軟件可靠性的主要措施,對于面向?qū)ο蟮能浖碚f,情況也是如此。面向?qū)ο鬁y試的目標,也是用盡可能低的測試成本發(fā)現(xiàn)盡可能多的軟件錯誤。但是,面向?qū)ο蟪绦蛑刑赜械姆庋b、繼承和多態(tài)等機制,也給面向?qū)ο鬁y試帶來一些新特點,增加了測試和調(diào)試的難度。必須在實踐中努力探索適合于面向?qū)ο筌浖母行У臏y試方法。面向?qū)ο笤O(shè)計的結(jié)果既可以用面向?qū)ο笳Z言、也可以用非面向?qū)ο笳Z言實現(xiàn)。12.1程序設(shè)計語言
12.1.1面向?qū)ο笳Z言的優(yōu)點使用面向?qū)ο笳Z言時,由于語言本身充分支持面向?qū)ο蟾拍畹膶崿F(xiàn),因此,編譯程序可以自動把面向?qū)ο蟾拍钣成涞侥繕顺绦蛑?。使用非面向?qū)ο笳Z言編寫面向?qū)ο蟪绦?,則必須由程序員自己把面向?qū)ο蟾拍钣成涞侥繕顺绦蛑?。所有非面向?qū)ο笳Z言都不支持一般-特殊結(jié)構(gòu)的實現(xiàn),使用這類語言編程時要么完全回避繼承的概念,要么在聲明特殊化類時,把對一般化類的引用嵌套在它里面。到底應(yīng)該選用面向?qū)ο笳Z言還是非面向?qū)ο笳Z言,關(guān)鍵不在于語言功能強弱。從原理上說,使用任何一種通用語言都可以實現(xiàn)面向?qū)ο蟾拍睢.斎?,使用面向?qū)ο笳Z言,實現(xiàn)面向?qū)ο蟾拍?,遠比使用非面向?qū)ο笳Z言方便,但是,方便性也并不是決定選擇何種語言的關(guān)鍵因素。選擇編程語言的關(guān)鍵因素,是語言的一致的表達能力、可重用性及可維護性。從面向?qū)ο笥^點看來,能夠更完整、更準確地表達問題域語義的面向?qū)ο笳Z言的語法是非常重要的,因為這會帶來下述幾個重要優(yōu)點:1.一致的表示方法從前面章節(jié)的講述中可以知道,面向?qū)ο箝_發(fā)基于不隨時間變化的、一致的表示方法。這種表示方法應(yīng)該從問題域到OOA,從OOA到OOD,最后從OOD到面向?qū)ο缶幊?OOP),始終穩(wěn)定不變。一致的表示方法既有利于在軟件開發(fā)過程中始終使用統(tǒng)一的概念,也有利于維護人員理解軟件的各種配置成分。2.可重用性為了能帶來可觀的商業(yè)利益,必須在更廣泛的范圍中運用重用機制,而不是僅僅在程序設(shè)計這個層次上進行重用。因此,在OOA,OOD直到OOP中都顯式地表示問題域語義,其意義是十分深遠的。隨著時間的推移,軟件開發(fā)組織既可能重用它在某個問題域內(nèi)的OOA結(jié)果,也可能重用相應(yīng)的OOD和OOP結(jié)果。3.可維護性盡管人們反復(fù)強調(diào)保持文檔與源程序一致的必要性,但是,在實際工作中很難做到交付兩類不同的文檔,并使它們保持彼此完全一致。特別是考慮到進度、預(yù)算、能力和人員等限制因素時,做到兩類文檔完全一致幾乎是不可能的。因此,維護人員最終面對的往往只有源程序本身。以ATM系統(tǒng)為例,說明在程序內(nèi)部表達問題域語義對維護工作的意義。假設(shè)在維護該系統(tǒng)時沒有合適的文檔資料可供參閱,于是維護人員人工瀏覽程序或使用軟件工具掃描程序,記下或打印出程序顯式陳述的問題域語義,維護人員看到“ATM”、“賬戶”、“現(xiàn)金兌換卡”等,這對維護人員理解所要維護的軟件將有很大幫助。因此,在選擇編程語言時,應(yīng)該考慮的首要因素,是在供選擇的語言中哪個語言能最好地表達問題域語義。一般說來,應(yīng)該盡量選用面向?qū)ο笳Z言來實現(xiàn)面向?qū)ο蠓治?、設(shè)計的結(jié)果。面向?qū)ο笳Z言的形成借鑒了歷史上許多程序語言的特點,從中吸取了豐富的營養(yǎng)。當今的面向?qū)ο笳Z言,從20世紀50年代誕生的LISP語言中引進了動態(tài)聯(lián)編的概念和交互式開發(fā)環(huán)境的思想,從20世紀60年代推出的SIMULA語言中引進了類的概念和繼承機制,此外,還受到20世紀70年代末期開發(fā)的Modula_2語言和Ada語言中數(shù)據(jù)抽象機制的影響。12.1.2面向?qū)ο笳Z言的技術(shù)特點20世紀80年代以來,面向?qū)ο笳Z言像雨后春筍一樣大量涌現(xiàn),形成了兩大類面向?qū)ο笳Z言。一類是純面向?qū)ο笳Z言,如Smalltalk和Eiffel等語言。另一類是混合型面向?qū)ο笳Z言,也就是在過程語言的基礎(chǔ)上增加面向?qū)ο髾C制,如C++等語言。一般說來,純面向?qū)ο笳Z言著重支持面向?qū)ο蠓椒ㄑ芯亢涂焖僭偷膶崿F(xiàn),而混合型面向?qū)ο笳Z言的目標則是提高運行速度和使傳統(tǒng)程序員容易接受面向?qū)ο笏枷?。成熟的面向?qū)ο笳Z言通常都提供豐富的類庫和強有力的開發(fā)環(huán)境。下面介紹在選擇面向?qū)ο笳Z言時應(yīng)該著重考察的一些技術(shù)特點。1.支持類與對象概念的機制所有面向?qū)ο笳Z言都允許用戶動態(tài)創(chuàng)建對象,并且可以用指針引用動態(tài)創(chuàng)建的對象。允許動態(tài)創(chuàng)建對象,就意味著系統(tǒng)必須處理內(nèi)存管理問題,如果不及時釋放不再需要的對象所占用的內(nèi)存,動態(tài)存儲分配就有可能耗盡內(nèi)存。有兩種管理內(nèi)存的方法,一種是由語言的運行機制自動管理內(nèi)存,即提供自動回收“垃圾”的機制;另一種是由程序員編寫釋放內(nèi)存的代碼。自動管理內(nèi)存不僅方便而且安全,但是必須采用先進的垃圾收集算法才能減少開銷。某些面向?qū)ο蟮恼Z言允許程序員定義析構(gòu)函數(shù)(destructor)。每當一個對象超出范圍或被顯式刪除時,就自動調(diào)用析構(gòu)函數(shù)。這種機制使得程序員能夠方便地構(gòu)造和喚醒釋放內(nèi)存的操作,卻又不是垃圾收集機制。2.實現(xiàn)整體-部分(即聚集)結(jié)構(gòu)的機制一般說來,有兩種實現(xiàn)方法,分別使用指針和獨立的關(guān)聯(lián)對象實現(xiàn)整體-部分結(jié)構(gòu)。大多數(shù)現(xiàn)有的面向?qū)ο笳Z言并不顯式支持獨立的關(guān)聯(lián)對象,在這種情況下,使用指針是最容易的實現(xiàn)方法,通過增加內(nèi)部指針可以方便地實現(xiàn)關(guān)聯(lián)。3.實現(xiàn)一般-特殊(即泛化)結(jié)構(gòu)的機制既包括實現(xiàn)繼承的機制也包括解決名字沖突的機制。所謂解決名字沖突,指的是處理在多個基類中可能出現(xiàn)的重名問題,這個問題僅在支持多重繼承的語言中才會遇到。某些語言拒絕接受有名字沖突的程序,另一些語言提供了解決沖突的協(xié)議。不論使用何種語言,程序員都應(yīng)該盡力避免出現(xiàn)名字沖突。4.實現(xiàn)屬性和服務(wù)的機制對于實現(xiàn)屬性的機制應(yīng)該著重考慮以下幾個方面:支持實例連接的機制;屬性的可見性控制;對屬性值的約束。對于服務(wù)來說,主要應(yīng)該考慮下列因素:支持消息連接(即表達對象交互關(guān)系)的機制;控制服務(wù)可見性的機制;動態(tài)聯(lián)編。所謂動態(tài)聯(lián)編,是指應(yīng)用系統(tǒng)在運行過程中,當需要執(zhí)行一個特定服務(wù)的時候,選擇(或聯(lián)編)實現(xiàn)該服務(wù)的適當算法的能力。動態(tài)聯(lián)編機制使得程序員在向?qū)ο蟀l(fā)送消息時擁有較大自由,在發(fā)送消息前,無須知道接受消息的對象當時屬于哪個類。5.類型檢查程序設(shè)計語言可以按照編譯時進行類型檢查的嚴格程度來分類。如果語言僅要求每個變量或?qū)傩噪`屬于一個對象,則是弱類型的;如果語法規(guī)定每個變量或?qū)傩员仨殰蚀_地屬于某個特定的類,則這樣的語言是強類型的。面向?qū)ο笳Z言在這方面差異很大,例如,Smalltalk實際上是一種無類型語言(所有變量都是未指定類的對象);C++和Eiffel則是強類型語言?;旌闲驼Z言(如C++,Objective_C等)甚至允許屬性值不是對象而是某種預(yù)定義的基本類型數(shù)據(jù)(如整數(shù),浮點數(shù)等),這可以提高操作的效率。強類型語言主要有兩個優(yōu)點:一是有利于在編譯時發(fā)現(xiàn)程序錯誤,二是增加了優(yōu)化的可能性。通常使用強類型編譯型語言開發(fā)軟件產(chǎn)品,使用弱類型解釋型語言快速開發(fā)原型??偟恼f來,強類型語言有助于提高軟件的可靠性和運行效率,現(xiàn)代的程序語言理論支持強類型檢查,大多數(shù)新語言都是強類型的。6.類庫大多數(shù)面向?qū)ο笳Z言都提供一個實用的類庫。某些語言本身并沒有規(guī)定提供什么樣的類庫,而是由實現(xiàn)這種語言的編譯系統(tǒng)自行提供類庫。存在類庫,許多軟構(gòu)件就不必由程序員重頭編寫了,這為實現(xiàn)軟件重用帶來很大方便。類庫中往往包含實現(xiàn)通用數(shù)據(jù)結(jié)構(gòu)(例如,動態(tài)數(shù)組、表、隊列、棧、樹等等)的類,通常把這些類稱為包容類。在類庫中還可以找到實現(xiàn)各種關(guān)聯(lián)的類。更完整的類庫通常還提供獨立于具體設(shè)備的接口類(例如,輸入輸出流),此外,用于實現(xiàn)窗口系統(tǒng)的用戶界面類也非常有用,它們構(gòu)成一個相對獨立的圖形庫。7.效率許多人認為面向?qū)ο笳Z言的主要缺點是效率低。產(chǎn)生這種印象的一個原因是,某些早期的面向?qū)ο笳Z言是解釋型的而不是編譯型的。事實上,使用擁有完整類庫的面向?qū)ο笳Z言,有時能比使用非面向?qū)ο笳Z言得到運行更快的代碼。這是因為類庫中提供了更高效的算法和更好的數(shù)據(jù)結(jié)構(gòu),例如,程序員已經(jīng)無須編寫實現(xiàn)哈希表或平衡樹算法的代碼了,類庫中已經(jīng)提供了這類數(shù)據(jù)結(jié)構(gòu),而且算法先進、代碼精巧可靠。認為面向?qū)ο笳Z言效率低的另一個理由是,這種語言在運行時使用動態(tài)聯(lián)編實現(xiàn)多態(tài)性,這似乎需要在運行時查找繼承樹,以得到定義給定操作的類。事實上,絕大多數(shù)面向?qū)ο笳Z言都優(yōu)化了這個查找過程,從而實現(xiàn)了高效率查找。只要在程序運行時始終保持類結(jié)構(gòu)不變,就能在子類中存儲各個操作的正確入口點,從而使得動態(tài)聯(lián)編成為查找哈希表的高效過程,不會由于繼承樹深度加大或類中定義的操作數(shù)增加而降低效率。8.持久保存對象任何應(yīng)用程序都對數(shù)據(jù)進行處理,如果希望數(shù)據(jù)能夠不依賴于程序執(zhí)行的生命期而長時間保存下來,則需要提供某種保存數(shù)據(jù)的方法。希望長期保存數(shù)據(jù)主要出于以下兩個原因:(1)為實現(xiàn)在不同程序之間傳遞數(shù)據(jù),需要保存數(shù)據(jù);(2)為恢復(fù)被中斷了的程序的運行,首先需要保存數(shù)據(jù)。一些面向?qū)ο笳Z言,沒有提供直接存儲對象的機制。這些語言的用戶必須自己管理對象的輸入輸出,或者購買面向?qū)ο蟮臄?shù)據(jù)庫管理系統(tǒng)。另外一些面向?qū)ο笳Z言(例如,Smalltalk),把當前的執(zhí)行狀態(tài)完整地保存在磁盤上。還有一些面向?qū)ο笳Z言,提供了訪問磁盤對象的輸入輸出操作。通過在類庫中增加對象存儲管理功能,可以在不改變語言定義或不增加關(guān)鍵字的情況下,就在開發(fā)環(huán)境中提供這種功能。然后,可以從“可存儲的類”中派生出需要持久保存的對象,該對象自然繼承了對象存儲管理功能。這就是Eiffel語言采用的策略。理想情況下,應(yīng)該使程序設(shè)計語言語法與對象存儲管理語法實現(xiàn)無縫集成。9.參數(shù)化類在實際的應(yīng)用程序中,常??吹竭@樣一些軟件元素(即函數(shù)、類等軟件成分),從它們的邏輯功能看,彼此是相同的,所不同的主要是處理的對象(數(shù)據(jù))類型不同。例如,對于一個向量(一維數(shù)組)類來說,不論是整型向量,浮點型向量,還是其他任何類型的向量,針對它的數(shù)據(jù)元素所進行的基本操作都是相同的(例如,插入、刪除、檢索等),當然,不同向量的數(shù)據(jù)元素的類型是不同的。如果程序語言提供一種能抽象出這類共性的機制,則對減少冗余和提高可重用性是大有好處的。所謂參數(shù)化類,就是使用一個或多個類型去參數(shù)化一個類的機制,有了這種機制,程序員就可以先定義一個參數(shù)化的類模板(即在類定義中包含以參數(shù)形式出現(xiàn)的一個或多個類型),然后把數(shù)據(jù)類型作為參數(shù)傳遞進來,從而把這個類模板應(yīng)用在不同的應(yīng)用程序中,或用在同一應(yīng)用程序的不同部分。Eiffel語言中就有參數(shù)化類,C++語言也提供了類模板。10.開發(fā)環(huán)境軟件工具和軟件工程環(huán)境對軟件生產(chǎn)率有很大影響。由于面向?qū)ο蟪绦蛑欣^承關(guān)系和動態(tài)聯(lián)編等引入的特殊復(fù)雜性,面向?qū)ο笳Z言所提供的軟件工具或開發(fā)環(huán)境就顯得尤其重要了。至少應(yīng)該包括下列一些最基本的軟件工具:編輯程序,編譯程序或解釋程序,瀏覽工具,調(diào)試器(debugger)等。編譯程序或解釋程序是最基本、最重要的軟件工具。編譯與解釋的差別主要是速度和效率不同。利用解釋程序解釋執(zhí)行用戶的源程序,雖然速度慢、效率低,但卻可以更方便更靈活地進行調(diào)試。編譯型語言適于用來開發(fā)正式的軟件產(chǎn)品,優(yōu)化工作做得好的編譯程序能生成效率很高的目標代碼。有些面向?qū)ο笳Z言(例如Objective_C)除提供編譯程序外,還提供一個解釋工具,從而給用戶帶來很大方便。某些面向?qū)ο笳Z言的編譯程序,先把用戶源程序翻譯成一種中間語言程序,然后再把中間語言程序翻譯成目標代碼。這樣做可能會使得調(diào)試器不能理解原始的源程序。在評價調(diào)試器時,首先應(yīng)該弄清楚它是針對原始的面向?qū)ο笤闯绦?,還是針對中間代碼進行調(diào)試。如果是針對中間代碼進行調(diào)試,則會給調(diào)試人員帶來許多不便。此外,面向?qū)ο蟮恼{(diào)試器,應(yīng)該能夠查看屬性值和分析消息連接的后果。在開發(fā)大型系統(tǒng)的時候,需要有系統(tǒng)構(gòu)造工具和變動控制工具。因此應(yīng)該考慮語言本身是否提供了這種工具,或者該語言能否與現(xiàn)有的這類工具很好地集成起來。經(jīng)驗表明,傳統(tǒng)的系統(tǒng)構(gòu)造工具(例如,UNIX的Make)目前對許多應(yīng)用系統(tǒng)來說都已經(jīng)太原始了。開發(fā)人員在選擇面向?qū)ο笳Z言時,還應(yīng)該著重考慮以下一些實際因素。1.將來能否占主導(dǎo)地位在若干年以后,哪種面向?qū)ο蟮某绦蛟O(shè)計語言將占主導(dǎo)地位呢?為了使自己的產(chǎn)品在若干年后仍然具有很強的生命力,人們可能希望采用將來占主導(dǎo)地位的語言編程。12.1.3選擇面向?qū)ο笳Z言根據(jù)目前占有的市場份額,以及專業(yè)書刊和學(xué)術(shù)會議上所做的分析、評價,人們往往能夠?qū)ξ磥砟姆N面向?qū)ο笳Z言將占據(jù)主導(dǎo)地位做出預(yù)測。但是,最終決定選用哪種面向?qū)ο笳Z言的實際因素,往往是諸如成本之類的經(jīng)濟因素而不是技術(shù)因素。2.可重用性采用面向?qū)ο蠓椒ㄩ_發(fā)軟件的基本目的和主要優(yōu)點,是通過重用提高軟件生產(chǎn)率。因此,應(yīng)該優(yōu)先選用能夠最完整、最準確地表達問題域語義的面向?qū)ο笳Z言。3.類庫和開發(fā)環(huán)境決定可重用性的因素,不僅僅是面向?qū)ο蟪绦蛘Z言本身,開發(fā)環(huán)境和類庫也是非常重要的因素。事實上,語言、開發(fā)環(huán)境和類庫這3個因素綜合起來,共同決定了可重用性??紤]類庫的時候,不僅應(yīng)該考慮是否提供了類庫,還應(yīng)該考慮類庫中提供了哪些有價值的類。隨著類庫的日益成熟和豐富,在開發(fā)新應(yīng)用系統(tǒng)時,需要開發(fā)人員自己編寫的代碼將越來越少。為便于積累可重用的類和重用已有的類,在開發(fā)環(huán)境中,除了提供前述的基本軟件工具外,還應(yīng)該提供使用方便的類庫編輯工具和瀏覽工具。其中的類庫瀏覽工具應(yīng)該具有強大的聯(lián)想功能。4.其他因素在選擇編程語言時,應(yīng)該考慮的其他因素還有:對用戶學(xué)習(xí)面向?qū)ο蠓治?、設(shè)計和編碼技術(shù)所能提供的培訓(xùn)服務(wù);在使用這個面向?qū)ο笳Z言期間能提供的技術(shù)支持;能提供給開發(fā)人員使用的開發(fā)工具、開發(fā)平臺、發(fā)行平臺;對機器性能和內(nèi)存的需求;集成已有軟件的容易程度等。在本書第7章已經(jīng)強調(diào)指出,良好的程序設(shè)計風格對保證程序質(zhì)量的重要性。良好的程序設(shè)計風格對面向?qū)ο髮崿F(xiàn)來說尤其重要,不僅能明顯減少維護或擴充的開銷,而且有助于在新項目中重用已有的程序代碼。良好的面向?qū)ο蟪绦蛟O(shè)計風格,既包括傳統(tǒng)的程序設(shè)計風格準則,也包括為適應(yīng)面向?qū)ο蠓椒ㄋ赜械母拍?例如,繼承性)而必須遵循的一些新準則。12.2程序設(shè)計風格面向?qū)ο蠓椒ǖ囊粋€主要目標,就是提高軟件的可重用性。軟件重用有多個層次,在編碼階段主要涉及代碼重用問題。一般說來,代碼重用有兩種:一種是本項目內(nèi)的代碼重用,另一種是新項目重用舊項目的代碼。內(nèi)部重用主要是找出設(shè)計中相同或相似的部分,然后利用繼承機制共享它們。為做到外部重用,則必須有長遠眼光,需要反復(fù)考慮精心設(shè)計。雖然為實現(xiàn)外部重用而需要考慮的面,比為實現(xiàn)內(nèi)部重用而需要考慮的面更廣,但是,有助于實現(xiàn)這兩類重用的程序設(shè)計準則卻是相同的。下面講述主要的準則:12.2.1提高可重用性1.提高方法的內(nèi)聚一個方法(即服務(wù))應(yīng)該只完成單個功能。如果某個方法涉及兩個或多個不相關(guān)的功能,則應(yīng)該把它分解成幾個更小的方法。2.減小方法的規(guī)模應(yīng)該減小方法的規(guī)模,如果某個方法規(guī)模過大(代碼長度超過一頁紙可能就太大了),則應(yīng)該把它分解成幾個更小的方法。3.保持方法的一致性保持方法的一致性,有助于實現(xiàn)代碼重用。一般說來,功能相似的方法應(yīng)該有一致的名字、參數(shù)特征(包括參數(shù)個數(shù)、類型和次序)、返回值類型、使用條件及出錯條件等。4.把策略與實現(xiàn)分開從所完成的功能看,有兩種不同類型的方法。一類方法負責做出決策,提供變元,并且管理全局資源,可稱為策略方法。另一類方法負責完成具體的操作,但卻并不做出是否執(zhí)行這個操作的決定,也不知道為什么執(zhí)行這個操作,可稱為實現(xiàn)方法。策略方法應(yīng)該檢查系統(tǒng)運行狀態(tài),并處理出錯情況,它們并不直接完成計算或?qū)崿F(xiàn)復(fù)雜的算法。策略方法通常緊密依賴于具體應(yīng)用,這類方法比較容易編寫,也比較容易理解。實現(xiàn)方法僅僅針對具體數(shù)據(jù)完成特定處理,通常用于實現(xiàn)復(fù)雜的算法。實現(xiàn)方法并不制定決策,也不管理全局資源,如果在執(zhí)行過程中發(fā)現(xiàn)錯誤,它們應(yīng)該只返回執(zhí)行狀態(tài)而不對錯誤采取行動。由于實現(xiàn)方法是自含式算法,相對獨立于具體應(yīng)用,因此,在其他應(yīng)用系統(tǒng)中也可能重用它們。為提高可重用性,在編程時不要把策略和實現(xiàn)放在同一個方法中,應(yīng)該把算法的核心部分放在一個單獨的具體實現(xiàn)方法中。為此需要從策略方法中提取出具體參數(shù),作為調(diào)用實現(xiàn)方法的變元。5.全面覆蓋如果輸入條件的各種組合都可能出現(xiàn),則應(yīng)該針對所有組合寫出方法,而不能僅僅針對當前用到的組合情況寫方法。例如,如果在當前應(yīng)用中需要寫一個方法,以獲取表中第一個元素,則至少還應(yīng)該為獲取表中最后一個元素再寫一個方法。此外,一個方法不應(yīng)該只能處理正常值,對空值、極限值及界外值等異常情況也應(yīng)該能夠作出有意義的響應(yīng)。6.盡量不使用全局信息應(yīng)該盡量降低方法與外界的耦合程度,不使用全局信息是降低耦合度的一項主要措施。7.利用繼承機制在面向?qū)ο蟪绦蛑校褂美^承機制是實現(xiàn)共享和提高重用程度的主要途徑。(1)調(diào)用子過程。最簡單的做法是把公共的代碼分離出來,構(gòu)成一個被其他方法調(diào)用的公用方法??梢栽诨愔卸x這個公用方法,供派生類中的方法調(diào)用,如圖12.1所示。圖12.1通過調(diào)用公用方法實現(xiàn)代碼重用(2)分解因子。有時提高相似類代碼可重用性的一個有效途徑,是從不同類的相似方法中分解出不同的“因子”(即不同的代碼),把余下的代碼作為公用方法中的公共代碼,把分解出的因子作為名字相同算法不同的方法,放在不同類中定義,并被這個公用方法調(diào)用,如圖12.2所示。使用這種途徑通常額外定義一個抽象基類,并在這個抽象基類中定義公用方法。把這種途徑與面向?qū)ο笳Z言提供的多態(tài)性機制結(jié)合起來,讓派生類繼承抽象基類中定義的公用方法,可以明顯降低為增添新子類而需付出的工作量,因為只需在新子類中編寫其特有的代碼。圖12.2通過因子分解實現(xiàn)代碼重用(3)使用委托。繼承關(guān)系的存在意味著子類“即是”父類,因此,父類的所有方法和屬性應(yīng)該都適用于子類。僅當確實存在一般-特殊關(guān)系時,使用繼承才是恰當?shù)?。繼承機制使用不當將造成程序難于理解、修改和擴充。當邏輯上不存在一般-特殊關(guān)系時,為重用已有的代碼,可以利用委托機制,如本書11.11.3小節(jié)所述。(4)把代碼封裝在類中。程序員往往希望重用用其他方法編寫的、解決同一類應(yīng)用問題的程序代碼。重用這類代碼的一個比較安全的途徑,是把被重用的代碼封裝在類中。例如,在開發(fā)一個數(shù)學(xué)分析應(yīng)用系統(tǒng)的過程中,已知有現(xiàn)成的實現(xiàn)矩陣變換的商品軟件包,程序員不想用C++語言重寫這個算法,于是他定義一個矩陣類把這個商品軟件包的功能封裝在該類中。上一小節(jié)所述的提高可重用性的準則,也能提高程序的可擴充性。此外,下列的面向?qū)ο蟪绦蛟O(shè)計準則也有助于提高可擴充性:1.封裝實現(xiàn)策略應(yīng)該把類的實現(xiàn)策略(包括描述屬性的數(shù)據(jù)結(jié)構(gòu)、修改屬性的算法等)封裝起來,對外只提供公有的接口,否則將降低今后修改數(shù)據(jù)結(jié)構(gòu)或算法的自由度。12.2.2提高可擴充性2.不要用一個方法遍歷多條關(guān)聯(lián)鏈一個方法應(yīng)該只包含對象模型中的有限內(nèi)容。違反這條準則將導(dǎo)致方法過分復(fù)雜,既不易理解,也不易修改擴充。3.避免使用多分支語句一般說來,可以利用DO_CASE語句測試對象的內(nèi)部狀態(tài),而不要用來根據(jù)對象類型選擇應(yīng)有的行為,否則在增添新類時將不得不修改原有的代碼。應(yīng)該合理地利用多態(tài)性機制,根據(jù)對象當前類型,自動決定應(yīng)有的行為。4.精心確定公有方法公有方法是向公眾公布的接口。對這類方法的修改往往會涉及許多其他類,因此,修改公有方法的代價通常都比較高。為提高可修改性,降低維護成本,必須精心選擇和定義公有方法。私有方法是僅在類內(nèi)使用的方法,通常利用私有方法來實現(xiàn)公有方法。刪除、增加或修改私有方法所涉及的面要窄得多,因此代價也比較低。程序員在編寫實現(xiàn)方法的代碼時,既應(yīng)該考慮效率,也應(yīng)該考慮健壯性。通常需要在健壯性與效率之間做出適當?shù)恼壑?。必須認識到,對于任何一個實用軟件來說,健壯性都是不可忽略的質(zhì)量指標。為提高健壯性應(yīng)該遵守以下幾條準則。12.2.3提高健壯性1.預(yù)防用戶的操作錯誤軟件系統(tǒng)必須具有處理用戶操作錯誤的能力。當用戶在輸入數(shù)據(jù)時發(fā)生錯誤,不應(yīng)該引起程序運行中斷,更不應(yīng)該造成“死機”。任何一個接收用戶輸入數(shù)據(jù)的方法,對其接收到的數(shù)據(jù)都必須進行檢查,即使發(fā)現(xiàn)了非常嚴重的錯誤,也應(yīng)該給出恰當?shù)奶崾拘畔ⅲ蕚湓俅谓邮沼脩舻妮斎搿?.檢查參數(shù)的合法性對公有方法,尤其應(yīng)該著重檢查其參數(shù)的合法性,因為用戶在使用公有方法時可能違反參數(shù)的約束條件。3.不要預(yù)先確定限制條件在設(shè)計階段,往往很難準確地預(yù)測出應(yīng)用系統(tǒng)中使用的數(shù)據(jù)結(jié)構(gòu)的最大容量需求。因此不應(yīng)該預(yù)先設(shè)定限制條件。如果有必要和可能,則應(yīng)該使用動態(tài)內(nèi)存分配機制,創(chuàng)建未預(yù)先設(shè)定限制條件的數(shù)據(jù)結(jié)構(gòu)。4.先測試后優(yōu)化為在效率與健壯性之間做出合理的折衷,應(yīng)該在為提高效率而進行優(yōu)化之前,先測試程序的性能,人們常常驚奇地發(fā)現(xiàn),事實上大部分程序代碼所消耗的運行時間并不多。應(yīng)該仔細研究應(yīng)用程序的特點,以確定哪些部分需要著重測試(例如,最壞情況出現(xiàn)的次數(shù)及處理時間,可能需要著重測試)。經(jīng)過測試,合理地確定為提高性能應(yīng)該著重優(yōu)化的關(guān)鍵部分。如果實現(xiàn)某個操作的算法有許多種,則應(yīng)該綜合考慮內(nèi)存需求、速度及實現(xiàn)的簡易程度等因素,經(jīng)合理折衷選定適當?shù)乃惴?。測試軟件的經(jīng)典策略是,從“小型測試”開始,逐步過渡到“大型測試”。用軟件測試的專業(yè)術(shù)語描述,就是從單元測試開始,逐步進入集成測試,最后進行確認測試和系統(tǒng)測試。對于傳統(tǒng)的軟件系統(tǒng)來說,單元測試集中測試最小的可編譯的程序單元(過程模塊),一旦把這些單元都測試完之后,就把它們集成到程序結(jié)構(gòu)中去;在集成過程中還應(yīng)該進行一系列的回歸測試,以發(fā)現(xiàn)模塊接口錯誤和新單元加入到程序中所帶來的副作用;最后,把軟件系統(tǒng)作為一個整體來測試,以發(fā)現(xiàn)軟件需求錯誤。測試面向?qū)ο筌浖牟呗耘c上述策略基本相同,但也有許多新特點。12.3測試策略當考慮面向?qū)ο蟮能浖r,單元的概念改變了?!胺庋b”導(dǎo)致了類和對象的定義,這意味著類和類的實例(對象)包裝了屬性(數(shù)據(jù))和處理這些數(shù)據(jù)的操作(也稱為方法或服務(wù))?,F(xiàn)在,最小的可測試單元是封裝起來的類和對象。一個類可以包含一組不同的操作,而一個特定的操作也可能存在于一組不同的類中。因此,對于面向?qū)ο蟮能浖碚f,單元測試的含義發(fā)生了很大變化。12.3.1面向?qū)ο蟮膯卧獪y試測試面向?qū)ο筌浖r,不能再孤立地測試單個操作,而應(yīng)該把操作作為類的一部分來測試。例如,假設(shè)有一個類層次,操作X在超類中定義并被一組子類繼承,每個子類都使用操作X,但是,X調(diào)用子類中定義的操作并處理子類的私有屬性。由于在不同的子類中使用操作X的環(huán)境有微妙的差別,因此有必要在每個子類的語境中測試操作X。這就說明,當測試面向?qū)ο筌浖r,傳統(tǒng)的單元測試方法是不適用的,不能再在“真空”中(即孤立地)測試單個操作。因為在面向?qū)ο蟮能浖胁淮嬖趯哟蔚目刂平Y(jié)構(gòu),傳統(tǒng)的自頂向下或自底向上的集成策略就沒有意義了。此外,由于構(gòu)成類的各個成分彼此間存在直接或間接的交互,一次集成一個操作到類中(傳統(tǒng)的漸增式集成方法)通常是不現(xiàn)實的。面向?qū)ο筌浖募蓽y試主要有下述兩種不同的策略。12.3.2面向?qū)ο蟮募蓽y試(1)基于線程的測試(threadbasedtesting)。這種策略把響應(yīng)系統(tǒng)的一個輸入或一個事件所需要的那些類集成起來。分別集成并測試每個線程,同時應(yīng)用回歸測試以保證沒有產(chǎn)生副作用。(2)基于使用的測試(usebasedtesting)。這種方法首先測試幾乎不使用服務(wù)器類的那些類(稱為獨立類),把獨立類都測試完之后,再測試使用獨立類的下一個層次的類(稱為依賴類)。對依賴類的測試一個層次一個層次地持續(xù)進行下去,直至把整個軟件系統(tǒng)構(gòu)造完為止。在測試面向?qū)ο蟮能浖^程中,應(yīng)該注意發(fā)現(xiàn)不同的類之間的協(xié)作錯誤。集群測試(clustertesting)是面向?qū)ο筌浖蓽y試的一個步驟。在這個測試步驟中,用精心設(shè)計的測試用例檢查一群相互協(xié)作的類(通過研究對象模型可以確定協(xié)作類),這些測試用例力圖發(fā)現(xiàn)協(xié)作錯誤。在確認測試或系統(tǒng)測試層次,不再考慮類之間相互連接的細節(jié)。和傳統(tǒng)的確認測試一樣,面向?qū)ο筌浖拇_認測試也集中檢查用戶可見的動作和用戶可識別的輸出。為了導(dǎo)出確認測試用例,測試人員應(yīng)該認真研究動態(tài)模型和描述系統(tǒng)行為的腳本,以確定最可能發(fā)現(xiàn)用戶交互需求錯誤的情景。當然,傳統(tǒng)的黑盒測試方法(見本書第7章)也可用于設(shè)計確認測試用例,但是,對于面向?qū)ο蟮能浖碚f,主要還是根據(jù)動態(tài)模型和描述系統(tǒng)行為的腳本來設(shè)計確認測試用例。12.3.3面向?qū)ο蟮拇_認測試目前,面向?qū)ο筌浖臏y試用例的設(shè)計方法,還處于研究、發(fā)展階段。與傳統(tǒng)軟件測試(測試用例的設(shè)計由軟件的輸入處理輸出視圖或單個模塊的算法細節(jié)驅(qū)動)不同,面向?qū)ο鬁y試關(guān)注于設(shè)計適當?shù)牟僮餍蛄幸詸z查類的狀態(tài)。12.4設(shè)計測試用例前面已經(jīng)講過,軟件測試從“小型測試”開始,逐步過渡到“大型測試”。對面向?qū)ο蟮能浖碚f,小型測試著重測試單個類和類中封裝的方法。測試單個類的方法主要有隨機測試、劃分測試和基于故障的測試等3種。12.4.1測試類的方法1.隨機測試下面通過銀行應(yīng)用系統(tǒng)的例子,簡要地說明這種測試方法。該系統(tǒng)的account(賬戶)類有下列操作:open(打開),setup(建立),deposit(存款),withdraw(取款),balance(余額),summarize(清單),creditLimit(透支限額)和close(關(guān)閉)。上列每個操作都可以應(yīng)用于account類的實例,但是,該系統(tǒng)的性質(zhì)也對操作的應(yīng)用施加了一些限制,例如,必須在應(yīng)用其他操作之前先打開賬戶,在完成了全部操作之后才能關(guān)閉賬戶。即使有這些限制,可做的操作也有許多種排列方法。一個account類實例的最小行為歷史包括下列操作:open·setup·deposit·withdraw·close這就是對account類的最小測試序列。但是,在下面的序列中可能發(fā)生許多其他行為:open·setup·deposit·[deposit|withdraw|balance|summarize|creditLimit]n·withdraw·close從上列序列可以隨機地產(chǎn)生一系列不同的操作序列,例如:測試用例#r1:open·setup·deposit·deposit·balance·summarize·withdraw·close測試用例#r2:open·setup·deposit·withdraw·deposit·balance·creditLimit·withdraw·close執(zhí)行上述這些及另外一些隨機產(chǎn)生的測試用例,可以測試類實例的不同生存歷史。2.劃分測試與測試傳統(tǒng)軟件時采用等價劃分方法類似,采用劃分測試(partitiontesting)方法可以減少測試類時所需要的測試用例的數(shù)量。首先,把輸入和輸出分類,然后設(shè)計測試用例以測試劃分出的每個類別。下面介紹劃分類別的方法。(1)基于狀態(tài)的劃分這種方法根據(jù)類操作改變類狀態(tài)的能力來劃分類操作。再一次考慮account類,狀態(tài)操作包括deposit和withdraw,而非狀態(tài)操作有balance,summarize和creditLimit。設(shè)計測試用例,以分別測試改變狀態(tài)的操作和不改變狀態(tài)的操作。例如,用這種方法可以設(shè)計出如下的測試用例:測試用例#p1:open·setup·deposit·deposit·withdraw·withdraw·close測試用例#p2:open·setup·deposit·summarize·creditLimit·withdraw·close測試用例#P1改變狀態(tài),而測試用例#P2測試不改變狀態(tài)的操作(在最小測試序列中的操作除外)。(2)基于屬性的劃分這種方法根據(jù)類操作使用的屬性來劃分類操作。對于account類來說,可以使用屬性balance來定義劃分,從而把操作劃分成3個類別:使用balance的操作;修改balance的操作;不使用也不修改balance的操作。然后,為每個類別設(shè)計測試序列。(3)基于功能的劃分這種方法根據(jù)類操作所完成的功能來劃分類操作。例如,可以把account類中的操作分類為初始化操作(open,setup),計算操作(deposit,withdraw),查詢操作(balance,summarize,creditLimit)和終止操作(close)。然后為每個類別設(shè)計測試序列。3.基于故障的測試基于故障的測試(faultbasedtesting)與傳統(tǒng)的錯誤推測法類似,也是首先推測軟件中可能有的錯誤,然后設(shè)計出最可能發(fā)現(xiàn)這些錯誤的測試用例。例如,軟件工程師經(jīng)常在問題的邊界處犯錯誤,因此,在測試SQRT(計算平方根)操作(該操作在輸入為負數(shù)時返回出錯信息)時,應(yīng)該著重檢查邊界情況:一個接近零的負數(shù)和零本身。其中“零本身”用于檢查程序員是否犯了如下錯誤:把語句if(x>=0)calculate_square_root();誤寫成if(x>0)calculate_square_root();為了推測出軟件中可能有的錯誤,應(yīng)該仔細研究分析模型和設(shè)計模型,而且在很大程度上要依靠測試人員的經(jīng)驗和直覺。如果推測得比較準確,則使用基于故障的測試方法能夠用相當?shù)偷墓ぷ髁堪l(fā)現(xiàn)大量錯誤;反之,如果推測不準,則這種方法的效果并不比隨機測試技術(shù)的效果好。開始集成面向?qū)ο笙到y(tǒng)以后,測試用例的設(shè)計變得更加復(fù)雜。在這個測試階段,必須對類間協(xié)作進行測試。為了舉例說明設(shè)計類間測試用例的方法,我們擴充12.4.1小節(jié)引入的銀行系統(tǒng)的例子,使它包含圖12.3所示的類和協(xié)作。圖中箭頭方向代表消息的傳遞方向,箭頭線上的標注給出了作為由消息所蘊含的協(xié)作的結(jié)果而調(diào)用的操作。和測試單個類相似,測試類協(xié)作可以使用隨機測試方法和劃分測試方法,以及基于情景的測試和行為測試來完成。12.4.2集成測試方法1.多類測試Kirani和Tsai建議使用下列步驟,以生成多個類的隨機測試用例。對每個客戶類,使用類操作符列表來生成一系列隨機測試序列。這些操作符向服務(wù)器類實例發(fā)送消息。對所生成的每個消息,確定協(xié)作類和在服務(wù)器對象中的對應(yīng)操作符。對服務(wù)器對象中的每個操作符(已經(jīng)被來自客戶對象的消息調(diào)用),確定傳遞的消息。對每個消息,確定下一層被調(diào)用的操作符,并把這些操作符結(jié)合進測試序列中。為了說明怎樣用上述步驟生成多個類的隨機測試用例,考慮Bank類相對于ATM類(見圖12.3)的操作序列:verifyAcct·verifyPIN·[(verifyPolicy·withdrawReq)|depositReq|acctInfoREQ]n對Bank類的隨機測試用例可能是:測試用例#r3:verifyAcct·verifyPIN·depositReq為了考慮在上述這個測試中涉及的協(xié)作者,需要考慮與測試用例#r3中的每個操作相關(guān)聯(lián)的消息。Bank必須和ValidationInfo協(xié)作以執(zhí)行verifyAcct和verifyPIN,Bank還必須和Account協(xié)作以執(zhí)行depositReq。因此,測試上面提到的協(xié)作的新測試用例是:測試用例#r4:verifyAcctBank·[validAcctValidationInfo]·verifyPINBank·[validPINvalidationInfo]·depositReq·[depositaccount]多個類的劃分測試方法類似于單個類的劃分測試方法(見12.4.1節(jié))。但是,對于多類測試來說,應(yīng)該擴充測試序列以包括那些通過發(fā)送給協(xié)作類的消息而被調(diào)用的操作。另一種劃分測試方法,根據(jù)與特定類的接口來劃分類操作。如圖12.3所示,Bank類接收來自ATM類和Cashier類的消息,因此,可以通過把Bank類中的方法劃分成服務(wù)于ATM的和服務(wù)于Cashier的兩類來測試它們。還可以用基于狀態(tài)的劃分(見12.4.1節(jié)),進一步精化劃分。圖12.3銀行系統(tǒng)的類-協(xié)作圖2.從動態(tài)模型導(dǎo)出測試用例在本書第9章中已經(jīng)講過,怎樣用狀態(tài)轉(zhuǎn)換圖作為表示類的動態(tài)行為的模型。類的狀態(tài)圖可以幫助我們導(dǎo)出測試該類(及與其協(xié)作的那些類)的動態(tài)行為的測試用例。圖12.4給出了前面討論過的account類的狀態(tài)圖,從圖可見,初始轉(zhuǎn)換經(jīng)過了emptyacct和setupacct這兩個狀態(tài),而類實例的大多數(shù)行為發(fā)生在workingacct狀態(tài)中,最終的withdraw和close使得account類分別向nonworkingacct狀態(tài)和deadacct狀態(tài)轉(zhuǎn)換。圖12.4account類的狀態(tài)轉(zhuǎn)換圖設(shè)計出的測
溫馨提示
- 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
- 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
- 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內(nèi)容里面會有圖紙預(yù)覽,若沒有圖紙預(yù)覽就沒有圖紙。
- 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
- 5. 人人文庫網(wǎng)僅提供信息存儲空間,僅對用戶上傳內(nèi)容的表現(xiàn)方式做保護處理,對用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對任何下載內(nèi)容負責。
- 6. 下載文件中如有侵權(quán)或不適當內(nèi)容,請與我們聯(lián)系,我們立即糾正。
- 7. 本站不保證下載資源的準確性、安全性和完整性, 同時也不承擔用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。
最新文檔
- 2025-2030年中國光學(xué)儀器行業(yè)商業(yè)模式創(chuàng)新戰(zhàn)略制定與實施研究報告
- 2025-2030年中國幼小銜接教育行業(yè)商業(yè)模式創(chuàng)新戰(zhàn)略制定與實施研究報告
- 2025-2030年中國旅游行業(yè)并購重組擴張戰(zhàn)略制定與實施研究報告
- 2025-2030年中國休閑餐飲行業(yè)全國市場開拓戰(zhàn)略制定與實施研究報告
- 2025-2030年中國知識密集型服務(wù)行業(yè)營銷創(chuàng)新戰(zhàn)略制定與實施研究報告
- 2025-2030年中國鉭電容器行業(yè)全國市場開拓戰(zhàn)略制定與實施研究報告
- 新形勢下智能門鎖行業(yè)轉(zhuǎn)型升級戰(zhàn)略制定與實施研究報告
- 德州黑陶品牌推廣調(diào)研
- 單位辦公室2025年工作要點
- 護肝藥品知識培訓(xùn)課件
- 梁平法制圖規(guī)則及鋼筋翻樣講解
- 乙肝 丙肝培訓(xùn)課件
- 2024屆湖北省武漢實驗外國語學(xué)校數(shù)學(xué)七上期末統(tǒng)考模擬試題含解析
- 基于深度學(xué)習(xí)的網(wǎng)絡(luò)釣魚郵件識別技術(shù)研究
- 融資成本視角下的船舶融資租賃模式研究
- 感冒中醫(yī)理論知識課件
- 2023年希望杯數(shù)學(xué)培訓(xùn)100題-六年級(含答案)
- 一年級科學(xué)人教版總結(jié)回顧2
- 個人住房貸款提前還款月供及節(jié)省利息EXCEL計算
- 第五單元《圓》教材解析-人教版數(shù)學(xué)六年級上冊
- 患者突發(fā)昏迷應(yīng)急預(yù)案演練腳本-
評論
0/150
提交評論