[TOC]
在這里簡單分享一下在Go中如何實現繼承。
1. 簡單的組合
說到繼承我們都知道,在Go中沒有extends關鍵字,也就意味著Go并沒有原生級別的繼承支持。這也是為什么我在文章開頭用了偽繼承這個詞璧榄。本質上特漩,Go使用interface實現的功能叫組合,Go是使用組合來實現的繼承骨杂,說的更精確一點涂身,是使用組合來代替的繼承,舉個很簡單的例子搓蚪。
1.1 實現父類
我們用很容易理解的動物-貓來舉例子蛤售,廢話不多說,直接看代碼妒潭。
type Animal struct {
Name string
}
func (a *Animal) Eat() {
fmt.Printf("%v is eating", a.Name)
fmt.Println()
}
type Cat struct {
*Animal
}
cat := &Cat{
Animal: &Animal{
Name: "cat",
},
}
cat.Eat() // cat is eating
1.2 代碼分析
首先悴能,我們實現了一個Animal的結構體,代表動物類雳灾。并聲明了Name字段漠酿,用于描述動物的名字。
然后谎亩,實現了一個以Animal為receiver的Eat方法炒嘲,來描述動物進食的行為。
最后匈庭,聲明了一個Cat結構體夫凸,組合了Cat字段。再實例化一個貓阱持,調用Eat方法夭拌,可以看到會正常的輸出。
可以看到衷咽,Cat結構體本身沒有Name字段鸽扁,也沒有去實現Eat方法。唯一有的就是組合了Animal父類兵罢,至此献烦,我們就證明了已經通過組合實現了繼承。
2. 優(yōu)雅的組合
上面的僅僅是為了給還沒有了解過Go組合的人看的卖词。作為一個簡單的例子來理解Go的組合繼承,這是完全沒有問題的 吏夯。但如果要運用在真正的開發(fā)中此蜈,那還是遠遠不夠的。
舉個例子噪生,我如果是這個抽象類的使用者裆赵,我拿到animal類不能一目了然的知道這個類干了什么,有哪些方法可以調用跺嗽。以及战授,沒有統(tǒng)一的初始化方式页藻,這意味著凡是涉及到初始化的地方都會有重復代碼。如果后期有初始化相關的修改植兰,那么只有一個一個挨著改份帐。所以接下來,我們對上述的代碼做一些優(yōu)化楣导。
2.1 抽象接口
接口用于描述某個類的行為废境。例如,我們即將要抽象的動物接口就會描述作為一個動物筒繁,具有哪些行為噩凹。常識告訴我們,動物可以進食(Eat)毡咏,可以發(fā)出聲音(bark)驮宴,可以移動(move)等等。這里有一個很有意思的類比呕缭。
// 模擬動物行為的接口
type IAnimal interface {
Eat() // 描述吃的行為
}
// 動物 所有動物的父類
type Animal struct {
Name string
}
// 動物去實現IAnimal中描述的吃的接口
func (a *Animal) Eat() {
fmt.Printf("%v is eating\n", a.Name)
}
// 動物的構造函數
func newAnimal(name string) *Animal {
return &Animal{
Name: name,
}
}
// 貓的結構體 組合了animal
type Cat struct {
*Animal
}
// 實現貓的構造函數 初始化animal結構體
func newCat(name string) *Cat {
return &Cat{
Animal: newAnimal(name),
}
}
cat := newCat("cat")
cat.Eat() // cat is eating
在Go中其實沒有關于構造函數的定義堵泽。例如我們在Java中可以使用構造函數來初始化變量,舉個很簡單的例子臊旭,Integer num = new Integer(1)落恼。而在Go中就需要使用者自己通過結構體的初始化來模擬構造函數的實現。
然后在這里我們實現子類Cat离熏,使用組合的方式代替繼承佳谦,來調用Animal中的方法。運行之后我們可以看到滋戳,Cat結構體中并沒有Name字段钻蔑,也沒有實現Eat方法,但是仍然可以正常運行奸鸯。這證明我們已經通過組合的方式了實現了繼承咪笑。
2.2 重寫方法
// 貓結構體IAnimal的Eat方法
func (cat *Cat) Eat() {
fmt.Printf("children %v is eating\n", cat.Name)
}
cat.Eat()
// children cat is eating
可以看到,Cat結構體已經重新實現了Animal中的Eat方法娄涩,這樣就實現了重寫窗怒。
2.3 參數多態(tài)
什么意思呢?舉個例子蓄拣,我們要如何在Java中解決函數的參數多態(tài)問題扬虚?熟悉Java的可能會想到一種解決方案,那就是通配符球恤。用一句話概括辜昵,使用了通配符可以使該函數接收某個類的所有父類型或者某個類的所有子類型。但是我個人認為對于不熟悉Java的人來說咽斧,可讀性不是特別友好堪置。
而在Go中躬存,就十分方便了。
func check(animal IAnimal) {
animal.Eat()
}
在這個函數中就可以處理所有組合了Animal的單位類型舀锨,對應到Java中就是上界通配符岭洲,即一個可以處理任何特定類型以及是該特定類型的派生類的通配符,再換句人話雁竞,啥動物都能處理钦椭。