Scala學習:使用組合與繼承
組合與繼承是利用其它現(xiàn)存類定義新類的兩個方法。如果你接下來的工作主要是代碼重用,通常你應(yīng)采用組合而不是繼承。只有繼承受脆基類問題之苦,這種情況你可能會無意中通過改變超類而破壞了子類。
51CTO編輯推薦:Scala編程語言專題
關(guān)于繼承關(guān)系你可以問自己一個問題,是否它建模了一個is-a關(guān)系。Meyers,《Effective C++》 【Mey91】例如,說ArrayElement是Element是合理的。你能問的另一個問題是,是否客戶想要把子類類型當作超類類型來用。Eckel,《Thinking in Java》【Eck98】在ArrayElement的例子里,我們的確期待客戶會想要把ArrayElement當作Element使用。
如果你對展示在圖釋10.3的繼承關(guān)系問了這些的問題,那么是否感覺其中的任何關(guān)系有可疑嗎?尤其是,對你來說LineElement是ArrayElement是否顯而易見呢?你是否認為客戶會需要把LineElement當作ArrayElement使用?實際上,我們把LineElement定義為ArrayElement主要是想重用ArrayElement的contents定義。因此或許把LineElement定義為Element的直接子類會更好一些,就像這樣:
前一個版本中,LineElement與ArrayElement有一個繼承關(guān)系,從那里繼承了contents?,F(xiàn)在它與Array有一個組合關(guān)系:在它自己的contents字段中持有一個字串數(shù)組的引用。類ArrayElement也與Array有組合關(guān)系,因為它的參數(shù)化contents字段持有字串數(shù)組的引用。ArrayElement的代碼展示在第xx頁的代碼10.5中。其組合關(guān)系用一個菱形表現(xiàn)在類圖中,正如展示在第xx頁的圖釋10.1中那樣。有了LineElement的這個實現(xiàn),Element的繼承層級現(xiàn)在看上去如展示在圖釋10.4中那樣。
- class LineElement(s: String) extends Element {
- val contents = Array(s)
- override def width = s.length
- override def height = 1
- }
圖釋 10.4 修改了LineElement后的類層級
【相關(guān)閱讀】