4.1 代碼復(fù)用的規(guī)則
代碼復(fù)用是絕大多數(shù)程序員所期望的,也是OO的目標(biāo)之一?偨Y(jié)我多年的編碼經(jīng)驗(yàn),為了使代碼能夠最大程度上復(fù)用,應(yīng)該特別注意以下幾個(gè)方面。
對(duì)接口編程
"對(duì)接口編程"是面向?qū)ο笤O(shè)計(jì)(OOD)的第一個(gè)基本原則。它的含義是:使用接口和同類(lèi)型的組件通訊,即,對(duì)于所有完成相同功能的組件,應(yīng)該抽象出一個(gè)接口,它們都實(shí)現(xiàn)該接口。具體到JAVA中,可以是接口(interface),或者是抽象類(lèi)(abstract class),所有完成相同功能的組件都實(shí)現(xiàn)該接口,或者從該抽象類(lèi)繼承。我們的客戶(hù)代碼只應(yīng)該和該接口通訊,這樣,當(dāng)我們需要用其它組件完成任務(wù)時(shí),只需要替換該接口的實(shí)現(xiàn),而我們代碼的其它部分不需要改變!
當(dāng)現(xiàn)有的組件不能滿(mǎn)足要求時(shí),我們可以創(chuàng)建新的組件,實(shí)現(xiàn)該接口,或者,直接對(duì)現(xiàn)有的組件進(jìn)行擴(kuò)展,由子類(lèi)去完成擴(kuò)展的功能。
優(yōu)先使用對(duì)象組合,而不是類(lèi)繼承
"優(yōu)先使用對(duì)象組合,而不是類(lèi)繼承"是面向?qū)ο笤O(shè)計(jì)的第二個(gè)原則。并不是說(shuō)繼承不重要,而是因?yàn)槊總(gè)學(xué)習(xí)OOP的人都知道OO的基本特性之一就是繼承,以至于繼承已經(jīng)被濫用了,而對(duì)象組合技術(shù)往往被忽視了。下面分析繼承和組合的優(yōu)缺點(diǎn):
類(lèi)繼承允許你根據(jù)其他類(lèi)的實(shí)現(xiàn)來(lái)定義一個(gè)類(lèi)的實(shí)現(xiàn)。這種通過(guò)生成子類(lèi)的復(fù)用通常被稱(chēng)為白箱復(fù)用(white-box reuse)。術(shù)語(yǔ)"白箱"是相對(duì)可視性而言:在繼承方式中,父類(lèi)的內(nèi)部細(xì)節(jié)對(duì)子類(lèi)可見(jiàn)。
對(duì)象組合是類(lèi)繼承之外的另一種復(fù)用選擇。新的更復(fù)雜的功能可以通過(guò)組合對(duì)象來(lái)獲得。對(duì)象組合要求對(duì)象具有良好定義的接口。這種復(fù)用風(fēng)格被稱(chēng)為黑箱復(fù)用(black-box reuse),因?yàn)楸唤M合的對(duì)象的內(nèi)部細(xì)節(jié)是不可見(jiàn)的。對(duì)象只以"黑箱"的形式出現(xiàn)。
繼承和組合各有優(yōu)缺點(diǎn)。類(lèi)繼承是在編譯時(shí)刻靜態(tài)定義的,且可直接使用,類(lèi)繼承可以較方便地改變父類(lèi)的實(shí)現(xiàn)。但是類(lèi)繼承也有一些不足之處。首先,因?yàn)槔^承在編譯時(shí)刻就定義了,所以無(wú)法在運(yùn)行時(shí)刻改變從父類(lèi)繼承的實(shí)現(xiàn)。更糟的是,父類(lèi)通常至少定義了子類(lèi)的部分行為,父類(lèi)的任何改變都可能影響子類(lèi)的行為。如果繼承下來(lái)的實(shí)現(xiàn)不適合解決新的問(wèn)題,則父類(lèi)必須重寫(xiě)或被其他更適合的類(lèi)替換。這種依賴(lài)關(guān)系限制了靈活性并最終限制了復(fù)用性。
對(duì)象組合是通過(guò)獲得對(duì)其他對(duì)象的引用而在運(yùn)行時(shí)刻動(dòng)態(tài)定義的。由于組合要求對(duì)象具有良好定義的接口,而且,對(duì)象只能通過(guò)接口訪問(wèn),所以我們并不破壞封裝性;只要類(lèi)型一致,運(yùn)行時(shí)刻還可以用一個(gè)對(duì)象來(lái)替代另一個(gè)對(duì)象;更進(jìn)一步,因?yàn)閷?duì)象的實(shí)現(xiàn)是基于接口寫(xiě)的,所以實(shí)現(xiàn)上存在較少的依賴(lài)關(guān)系。
優(yōu)先使用對(duì)象組合有助于你保持每個(gè)類(lèi)被封裝,并且只集中完成單個(gè)任務(wù)。這樣類(lèi)和類(lèi)繼承層次會(huì)保持較小規(guī)模,并且不太可能增長(zhǎng)為不可控制的龐然大物(這正是濫用繼承的后果)。另一方面,基于對(duì)象組合的設(shè)計(jì)會(huì)有更多的對(duì)象(但只有較少的類(lèi)),且系統(tǒng)的行為將依賴(lài)于對(duì)象間的關(guān)系而不是被定義在某個(gè)類(lèi)中。
注意:理想情況下,我們不用為獲得復(fù)用而去創(chuàng)建新的組件,只需要使用對(duì)象組合技術(shù),通過(guò)組裝已有的組件就能獲得需要的功能。但是事實(shí)很少如此,因?yàn)榭捎玫慕M件集合并不豐富。使用繼承的復(fù)用使得創(chuàng)建新的組件要比組裝已有的組件來(lái)得容易。這樣,繼承和對(duì)象組合常一起使用。然而,正如前面所說(shuō),千萬(wàn)不要濫用繼承而忽視了對(duì)象組合技術(shù)。
相關(guān)推薦:計(jì)算機(jī)等級(jí)考試二級(jí)Java經(jīng)典算法大全匯總
北京 | 天津 | 上海 | 江蘇 | 山東 |
安徽 | 浙江 | 江西 | 福建 | 深圳 |
廣東 | 河北 | 湖南 | 廣西 | 河南 |
海南 | 湖北 | 四川 | 重慶 | 云南 |
貴州 | 西藏 | 新疆 | 陜西 | 山西 |
寧夏 | 甘肅 | 青海 | 遼寧 | 吉林 |
黑龍江 | 內(nèi)蒙古 |