“這就是問題了, 編譯期/源代碼的依賴導(dǎo)致我們沒法把系統(tǒng)劃分成獨立的組件,各自獨立開發(fā),獨立部署,一個的變化就影響到了另外一個。”
Java說:“你看看我使用多態(tài)以后的設(shè)計,我的業(yè)務(wù)邏輯在編譯時只依賴那個接口Writer,而不依賴具體的實現(xiàn)FileWriter和MailWriter。”
為什么面向?qū)ο笤阃噶耍?/div>
“你的意思是只要接口Writer不變化,底層的具體實現(xiàn)如FileWriter ,MailWriter可以隨意變化,隨意替換,就像插件一樣,對吧?” C語言說到。
“對啊,編譯期/源代碼不依賴,運行期依賴,這就是延遲綁定帶來的好處,現(xiàn)在你明白o.f()和 f(o)的本質(zhì)區(qū)別了吧。 ”
Lisp 不失時機又插了一嘴:“你那接口中只有一個函數(shù),就是write(),用什么接口啊,脫褲子放屁,多此一舉,在我這兒只要把不同的函數(shù)傳遞過去就可以了。”
Java笑道:“別抬杠,這就是個簡單的例子,不管是用接口,還是傳遞函數(shù),都是延遲綁定嘛, 關(guān)鍵點都是要找到那個穩(wěn)定的東西(Writer),就是抽象。你找不到這個穩(wěn)定的東西,做不出抽象,你的系統(tǒng)就沒法劃分成可以獨立開發(fā),獨立變化的組件了。”
C語言還想反擊,但一直找不到突破口。
Lisp說道:“別聽Java在哪里忽悠,C老弟,你也能實現(xiàn)運行期的延遲綁定,這不是Java的專利,你忘了虛函數(shù)表了?”
C語言一拍大腿:“是啊,我忘了那一年春節(jié)回家,Linus 大神曾經(jīng)告訴我,虛函數(shù)表和函數(shù)指針才是實現(xiàn)多態(tài)的關(guān)鍵,比如Unix/Linux把設(shè)備都當(dāng)成了文件,有標(biāo)準(zhǔn)的open , read 等方法,對于不同的設(shè)備,都能調(diào)用對應(yīng)的方法, 那是怎么實現(xiàn)的?也是通過虛函數(shù)表做延遲綁定嘛!”
(碼農(nóng)翻身老劉注:C語言實現(xiàn)面向?qū)ο蟮脑斍榭梢砸撇健?
C語言高興了:“哈哈,Java 老弟,看來我們本質(zhì)上還是一樣的,多態(tài)只不過是函數(shù)指針的一種應(yīng)用!”
Java說:“所以編程的關(guān)鍵不在于是否使用了面向?qū)ο蟮恼Z言,這一點你同意吧?”
C語言點頭,編程的關(guān)鍵點就是找到、抽象出穩(wěn)定的接口,針對這個接口編程,這樣就可以讓各個模塊能夠獨立地變化。
“說起來容易,做起來難,這兒有一個例子,你給我用面向?qū)ο笤O(shè)計一下?” Lisp 拋出了一道題。
動物可以分為肉食動物,草食動物,水生動物,陸生動物,用類如何表示?
Java 說:這還不簡單,看看這名詞多明顯啊,都可以變成類啊,讓他們都繼承動物就可以了。
為什么面向?qū)ο笤阃噶耍?/div>
Lisp看到Java 掉入了陷阱,狡黠地一笑:“那有的動物就既是陸生動物,又是肉食動物,怎么表達(dá)?”
“那我就加一個陸生肉食動物類。” 雖然覺得不妥,Java還是說了出來。
為什么面向?qū)ο笤阃噶耍?/div>
“那要是再來一個水生肉食動物怎么辦?或者來了一個新的概念‘哺乳動物’, 該怎么處理?”
“哈哈哈,我懂了,隨著需求的增多,不但類會出現(xiàn)爆炸, 還可能會出現(xiàn)那種怪異的類,這面向?qū)ο缶幊檀_實是有大問題啊!” C語言說。
Java 低頭沉思不語,突然,腦海中想起來了那句話:優(yōu)先使用組合而不是繼承。
怎么使用組合?必須得改變下看待問題的方式,對,應(yīng)該這樣: