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語言中,如果子類包含了父類的結構,我們就能直接用子類「.」出父類的函數。這不得不說是一個偉大的特性。它讓我們的「繼承」更加天衣無縫了。

總結:通過模版模式,我們可以把子類做為父類的模版,提取出公共的結構到父類,共享父類的程式碼。這樣能消除程式碼結構重複的壞味。並且,簡化了子類的功能,使之職責單一的為「父類」提供數據。