go語言實現設計模式(三):模版模式
- 2019 年 11 月 21 日
- 筆記
模版模式真的是一個好東西。所謂模版模式,就是說,某幾個類中相同的操作和程式碼提取到父類的一個函數中,並定義相同的操作為抽象函數。由子類來實現。估計我也沒表達清楚,下面還是看程式碼來講解吧。
例:我們有兩個文檔處理程式,一個處理本地數據,一個處理網路數據。這兩個類的功能和執行步驟高度類似,但是,獲取數據的方式不同。在這兩個類中勢必存在大量結構相同的程式碼。現在,我們使用模版模式來重構這兩個類。
1.定義父類,在父類中定義實現的具體函數和一個等待子類實例化的「抽象函數」
package doc import "fmt" type DocSuper struct { GetContent func() string } func (d DocSuper) DoOperate() { fmt.Println("對這個文檔做了一些處理,文檔是:", d.GetContent()) }
我們把這兩個類,提取出一個父類,DocSuper。這個類中有兩個函數,一個具體的函數,DoOperate,用來做實際的操作。一個抽象函數,等待子類實現,用來獲取不同文檔類型的內容。其實,我們只是等待子類實現這個抽象函數,好為父類的DoOperate來提供數據:)。
另外,我們最最好用的go語言,是沒有抽象類,抽象函數之類的概念的,所以我使用了一個指向一個函數的指針(func() string)來模擬實現抽象函數。如果我們的子類需要「實例化」這個「抽象函數」(實質是子類給這個父類函數指針賦值),就必須滿足我們的指針約束。
2.定義子類
package doc type LocalDoc struct { DocSuper } func NewLocalDoc() *LocalDoc { c := new(LocalDoc) c.DocSuper.GetContent = c.GetContent return c } func (e *LocalDoc) GetContent() string { return "this is a LocalDoc." }
package doc type NetDoc struct { DocSuper } func NewNetDoc() *NetDoc { c := new(NetDoc) c.DocSuper.GetContent = c.GetContent return c } func (c *NetDoc) GetContent() string { return "this is net doc." }
大家仔細看程式碼就能發現,我們的子類中可以包含父類的結構,雖然go語言不能繼承,但是通過這種方式,我們也能過一過「繼承」的癮。當我們使用NewNetDoc函數實例化的時候,在這個函數中,會把子類的函數指針賦值給父類的函數指針。此時,子類中包含了父類的結構,故而我們能調用父類的函數來完成功能。
3.接下來就是測試了
// 模版模式 project main.go package main import . "doc" func main() { netDoc := NewNetDoc() lcDoc := NewLocalDoc() netDoc.DoOperate() lcDoc.DoOperate() } /* 輸出: 對這個文檔做了一些處理,文檔是: this is net doc. 對這個文檔做了一些處理,文檔是: this is a LocalDoc. */
在go語言中,如果子類包含了父類的結構,我們就能直接用子類「.」出父類的函數。這不得不說是一個偉大的特性。它讓我們的「繼承」更加天衣無縫了。
總結:通過模版模式,我們可以把子類做為父類的模版,提取出公共的結構到父類,共享父類的程式碼。這樣能消除程式碼結構重複的壞味。並且,簡化了子類的功能,使之職責單一的為「父類」提供數據。