這是一個建立於 的文章,其中的資訊可能已經有所發展或是發生改變。
模版模式真的是一個好東西。所謂模版模式,就是說,某幾個類中相同的操作和代碼提取到父類的一個函數中,並定義相同的操作為抽象函數。由子類來實現。估計我也沒表達清楚,下面還是看代碼來講解吧。
例:我們有兩個文檔處理常式,一個處理本機資料,一個處理網路資料。這兩個類的功能和執行步驟高度類似,但是,擷取資料的方式不同。在這兩個類中勢必存在大量結構相同的代碼。現在,我們使用模版模式來重構這兩個類。
1.定義父類,在父類中定義實現的具體函數和一個等待子類執行個體化的“抽象函數”
package docimport "fmt"type DocSuper struct { GetContent func() string}func (d DocSuper) DoOperate() { fmt.Println("對這個文檔做了一些處理,文檔是:", d.GetContent())}
我們把這兩個類,提取出一個父類,DocSuper。這個類中有兩個函數,一個具體的函數,DoOperate,用來做實際的操作。一個抽象函數,等待子類實現,用來擷取不同文件類型的內容。其實,我們只是等待子類實現這個抽象函數,好為父類的DoOperate來提供資料:)。
另外,我們最最好用的go語言,是沒有抽象類別,抽象函數之類的概念的,所以我使用了一個指向一個函數的指標(func() string)來類比實現抽象函數。如果我們的子類需要“執行個體化”這個“抽象函數”(實質是子類給這個父類函數指標賦值),就必須滿足我們的指標約束。
2.定義子類
package doctype 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 doctype 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.gopackage mainimport . "doc"func main() { netDoc := NewNetDoc() lcDoc := NewLocalDoc() netDoc.DoOperate() lcDoc.DoOperate() }/*輸出:對這個文檔做了一些處理,文檔是: this is net doc.對這個文檔做了一些處理,文檔是: this is a LocalDoc.*/
在go語言中,如果子類包含了父類的結構,我們就能直接用子類“.”出父類的函數。這不得不說是一個偉大的特性。它讓我們的“繼承”更加天衣無縫了。
總結:通過模版模式,我們可以把子類做為父類的模版,提取出公用的結構到父類,共用父類的代碼。這樣能消除代碼結構重複的壞味。並且,簡化了子類的功能,使之職責單一的為“父類”提供資料。
著作權聲明:本文為博主原創文章,未經博主允許不得轉載。