Go 分布式学习利器(12)-- Go语言的扩展和复用
Go語(yǔ)言無(wú)法天然支持繼承,但是又想要實(shí)現(xiàn)面向?qū)ο蟮奶匦浴?br /> 即父類(lèi)對(duì)象 使用子類(lèi)對(duì)象初始化,那么該父類(lèi)對(duì)象調(diào)用的函數(shù)就是子類(lèi)實(shí)現(xiàn)的函數(shù) ,從而滿(mǎn)足LSP(子類(lèi)交換原則)。
案例一: Go語(yǔ)言 支持?jǐn)U展父類(lèi)的功能,如下代碼:
package oriented_testimport ("fmt""testing"
)// Pet 類(lèi)
type Pet struct {
}func (p *Pet) Speak(){ // Pet類(lèi)的函數(shù)成員fmt.Print("Pet speak.\n")
}func (p *Pet) SpeakTo(host string) { // Pet類(lèi)的函數(shù)成員p.Speak()fmt.Println("Pet SpeakTo ", host)
}// Dog 擴(kuò)展Pet的功能
type Dog struct {p *Pet
}// 擴(kuò)展父類(lèi)的方法
func (d *Dog) Speak(){d.p.Speak()
}// 擴(kuò)展父類(lèi)的方法
func (d *Dog) SpeakTo(host string) {d.Speak()fmt.Println("Dog Speakto ", host)
}func TestDog(t *testing.T) {dog := new(Dog)dog.SpeakTo("Test dog")
}
以上測(cè)試代碼的輸出如下:
Pet speak.
Dog Speakto Test dog
dog的SpeakTo中調(diào)用了dog 的Speak,其中調(diào)用了Pet的Speak,所以輸出正常。
Pet 和 Dog調(diào)用不會(huì)相互影響,完全由用戶(hù)決定。
但是這和我們所想需要的不同,Pet類(lèi)有自己的方法,Dog類(lèi)有自己的方法,兩者作用域完全不同。
這里Go語(yǔ)言推出了匿名嵌套類(lèi)型,即Dog類(lèi)不用實(shí)現(xiàn)自己的和Pet類(lèi)同名的方法即可,通過(guò)在Dog類(lèi)的聲明中變更Pet成員。
案例二: Go語(yǔ)言支持匿名函數(shù)類(lèi)型
// Dog 擴(kuò)展Pet的功能
type Dog struct {Pet
}
這樣即不需要Dog聲明自己的同名函數(shù)成員,默認(rèn)的調(diào)用即為Pet成員函數(shù)的調(diào)用
package oriented_testimport ("fmt""testing"
)type Pet struct {
}func (p *Pet) Speak(){fmt.Print("Pet speak.\n")
}func (p *Pet) SpeakTo(host string) {p.Speak()fmt.Println("Pet SpeakTo ", host)
}// Dog 擴(kuò)展Pet的功能
type Dog struct {Pet // 支持匿名嵌套類(lèi)型,
}func TestDog(t *testing.T) {var dog Dogdog.Speak()dog.SpeakTo("Test dog")
}
最終的輸出如下:
=== RUN TestDog
Pet speak.
Pet speak.
Pet SpeakTo Test dog
--- PASS: TestDog (0.00s)
調(diào)用的都是Pet的成員函數(shù),感覺(jué)像是繼承了,因?yàn)槔^承默認(rèn)就是子類(lèi)能夠使用父類(lèi)的公有成員。
在匿名嵌套類(lèi)型下,我們想要完整嘗試一下Go語(yǔ)言是否真正支持繼承,可以像之前的代碼一樣在Dog中實(shí)現(xiàn)Pet的同名函數(shù),且能夠通過(guò)父類(lèi)對(duì)象調(diào)用子類(lèi)的成員方法,像C++/Java這樣進(jìn)行向上類(lèi)型轉(zhuǎn)換(本身是不可能的,Go語(yǔ)言不支持顯式類(lèi)型轉(zhuǎn)換)。
案例三: Go語(yǔ)言不支持繼承,如下代碼:
package oriented_testimport ("fmt""testing"
)type Pet struct {
}func (p *Pet) Speak(){fmt.Print("Pet speak.\n")
}func (p *Pet) SpeakTo(host string) {p.Speak()fmt.Println("Pet SpeakTo ", host)
}// Dog 擴(kuò)展Pet的功能
type Dog struct {//p *PetPet // 支持匿名嵌套類(lèi)型
}// 重載父類(lèi)的方法
func (d *Dog) Speak(){fmt.Print("Dog speak.\n")
}// 重載父類(lèi)的方法
func (d *Dog) SpeakTo(host string) {d.Speak()fmt.Println("Dog Speakto ", host)
}func TestDog(t *testing.T) {var dog Pet = new(Dog)dog.Speak()dog.SpeakTo("Test dog")
}
在最后的輸出會(huì)編譯出錯(cuò),不支持將Pet類(lèi)型轉(zhuǎn)換為Dog類(lèi)型:
./oriented_test.go:38:6: cannot use new(Dog) (type *Dog) as type Pet in assignment
總結(jié)一下,Go語(yǔ)言并不支持繼承,能夠支持接口的擴(kuò)展 和 復(fù)用(匿名嵌套類(lèi)型)
其中擴(kuò)展 就是不同類(lèi)實(shí)現(xiàn)相同的成員函數(shù),能夠?qū)崿F(xiàn)類(lèi)似于案例一中的擴(kuò)展接口形態(tài)。
復(fù)用則是通過(guò)匿名嵌套類(lèi)型實(shí)現(xiàn) 類(lèi)似于重載的功能,可以看看案例二的代碼。
總結(jié)
以上是生活随笔為你收集整理的Go 分布式学习利器(12)-- Go语言的扩展和复用的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: 女性不孕不育一般多久能治好
- 下一篇: Go 分布式学习利器(13)-- Go语