為什么阿里巴巴建議開(kāi)發(fā)者謹(jǐn)慎使用繼承?
從學(xué)習(xí)Java的第一天起,我們就知道Java是一種面向?qū)ο笳Z(yǔ)言,而學(xué)習(xí)Java的第二天,我們就知道了面向?qū)ο蟮娜蠡咎匦允牵悍庋b、繼承、多態(tài)。所以,對(duì)于很多開(kāi)發(fā)者來(lái)說(shuō),繼承肯定都是不陌生的。但是,繼承一定適合所有的場(chǎng)景嗎?毫無(wú)忌諱的使用繼承來(lái)做代碼擴(kuò)展真的好嗎?為什么《阿里巴巴Java開(kāi)發(fā)手冊(cè)》中有一條規(guī)定:謹(jǐn)慎使用繼承的方式進(jìn)行擴(kuò)展,優(yōu)先使用組合的方式實(shí)現(xiàn)。
本文就來(lái)針對(duì)這些問(wèn)題,簡(jiǎn)單分析一下。
1.面向?qū)ο蟮膹?fù)用技術(shù)
每個(gè)人在剛剛學(xué)習(xí)繼承的時(shí)候都會(huì)或多或少的有這樣一個(gè)印象:繼承可以幫助我實(shí)現(xiàn)類(lèi)的復(fù)用。所以,很多開(kāi)發(fā)人員在需要復(fù)用一些代碼的時(shí)候會(huì)很自然的使用類(lèi)的繼承的方式,因?yàn)闀?shū)上就是這么寫(xiě)的(老師就是這么教的)。但是,其實(shí)這樣做是不對(duì)的。長(zhǎng)期大量的使用繼承會(huì)給代碼帶來(lái)很高的維護(hù)成本。前面提到復(fù)用,這里就簡(jiǎn)單介紹一下面向?qū)ο蟮膹?fù)用技術(shù)。復(fù)用性是面向?qū)ο蠹夹g(shù)帶來(lái)的很棒的潛在好處之一。如果運(yùn)用的好的話(huà)可以幫助我們節(jié)省很多開(kāi)發(fā)時(shí)間,提升開(kāi)發(fā)效率。但是,如果被濫用那么就可能產(chǎn)生很多難以維護(hù)的代碼。作為一門(mén)面向?qū)ο箝_(kāi)發(fā)的語(yǔ)言,代碼復(fù)用是Java引人注意的功能之一。Java代碼的復(fù)用有繼承,組合以及代理三種具體的表現(xiàn)形式。
2.繼承
繼承(Inheritance)是一種聯(lián)結(jié)類(lèi)與類(lèi)的層次模型。指的是一個(gè)類(lèi)(稱(chēng)為子類(lèi)、子接口)繼承另外的一個(gè)類(lèi)(稱(chēng)為父類(lèi)、父接口)的功能,并可以增加它自己的新功能的能力,繼承是類(lèi)與類(lèi)或者接口與接口之間最常見(jiàn)的關(guān)系。
繼承是一種is-a關(guān)系。如蘋(píng)果是水果,狗是動(dòng)物,哈士奇是狗。
3.組合
組合(Composition)體現(xiàn)的是整體與部分、擁有的關(guān)系。
組合是一種has-a的關(guān)系。如汽車(chē)有一個(gè)發(fā)動(dòng)機(jī),學(xué)校有一個(gè)老師等。
4.組合與繼承的區(qū)別
首先,從類(lèi)的關(guān)系確定時(shí)間點(diǎn)上,組合和繼承是有區(qū)別的:繼承,在寫(xiě)代碼的時(shí)候就要指名具體繼承哪個(gè)類(lèi),所以,類(lèi)的繼承關(guān)系是在編譯期就確定的。并且從基類(lèi)繼承來(lái)的實(shí)現(xiàn)是無(wú)法在運(yùn)行期動(dòng)態(tài)改變的,因此降低了應(yīng)用的靈活性。組合,在寫(xiě)代碼的時(shí)候可以采用面向接口編程。所以,類(lèi)的組合關(guān)系一般在運(yùn)行期確定。另外,代碼復(fù)用方式上也有一定區(qū)別:繼承結(jié)構(gòu)中,父類(lèi)的內(nèi)部細(xì)節(jié)對(duì)于子類(lèi)是可見(jiàn)的。所以我們通常也可以說(shuō)通過(guò)繼承的代碼復(fù)用是一種白盒式代碼復(fù)用。如果基類(lèi)的實(shí)現(xiàn)發(fā)生改變,那么派生類(lèi)的實(shí)現(xiàn)也將隨之改變。這樣就導(dǎo)致了子類(lèi)行為的不可預(yù)知性。組合是通過(guò)對(duì)現(xiàn)有的對(duì)象進(jìn)行拼裝(組合)產(chǎn)生新的、更復(fù)雜的功能。因?yàn)樵趯?duì)象之間,各自的內(nèi)部細(xì)節(jié)是不可見(jiàn)的,所以我們也說(shuō)通過(guò)組合的代碼復(fù)用是黑盒式代碼復(fù)用。因?yàn)榻M合中一般都定義一個(gè)類(lèi)型,所以在編譯期根本不知道具體會(huì)調(diào)用哪個(gè)實(shí)現(xiàn)類(lèi)的方法。最后,Java中不支持多繼承,而組合是沒(méi)有限制的。就像一個(gè)人只能有一個(gè)父親,但是他可以有很很多輛車(chē)。
5.優(yōu)缺點(diǎn)對(duì)比
6.為什么組合優(yōu)于繼承
相信很多人都知道面向?qū)ο笾杏幸粋€(gè)比較重要的原則『多用組合、少用繼承』或者說(shuō)『組合優(yōu)于繼承』。從前面的介紹已經(jīng)優(yōu)缺點(diǎn)對(duì)比中也可以看出,組合比繼承更加靈活,也更有助于代碼維護(hù)。其具有不破壞封裝性、具有更好的可擴(kuò)展性、支持動(dòng)態(tài)組合、整體類(lèi)可以改變局部類(lèi)的行為等優(yōu)點(diǎn)。所以,建議在同樣可行的情況下,優(yōu)先使用組合而不是繼承。因?yàn)榻M合更安全,更簡(jiǎn)單,更靈活,更高效。注意,并不是說(shuō)繼承就一點(diǎn)用都沒(méi)有了,前面說(shuō)的是【在同樣可行的情況下】。有一些場(chǎng)景還是需要使用繼承的,或者是更適合使用繼承。
【本文是51CTO專(zhuān)欄作者Hollis的原創(chuàng)文章,作者微信公眾號(hào)Hollis(ID:hollischuang)】