嵌入類型,或者嵌套類型,這是一種可以把已有的類型聲明在新的類型里的一種方式,這種功能對代碼復(fù)用非常重要。
創(chuàng)新互聯(lián)自2013年創(chuàng)立以來,是專業(yè)互聯(lián)網(wǎng)技術(shù)服務(wù)公司,擁有項目做網(wǎng)站、網(wǎng)站設(shè)計網(wǎng)站策劃,項目實施與項目整合能力。我們以讓每一個夢想脫穎而出為使命,1280元新興做網(wǎng)站,已為上家服務(wù),為新興各地企業(yè)和個人服務(wù),聯(lián)系電話:18980820575
在其他語言中,有繼承可以做同樣的事情,但是在Go語言中,沒有繼承的概念。Go提倡的代碼復(fù)用的方式是組合,所以這也是嵌入類型的意義所在。組合而不是繼承,所以Go才會更靈活。
type Reader interface { Read(p []byte) (n int, err error) } type Writer interface { Write(p []byte) (n int, err error) } type Closer interface { Close() error } type ReadWriter interface { Reader Writer } type ReadCloser interface { Reader Closer } type WriteCloser interface { Writer Closer }
以上是標(biāo)準(zhǔn)庫io
包里我們常用的接口,可以看到ReadWriter
接口是嵌入Reader
和Reader
接口而組合成的新接口,這樣我們就不用重復(fù)地定義被嵌入接口里的方法,直接通過嵌入就可以了。嵌入類型同樣適用于結(jié)構(gòu)體類型,我們再來看個例子:
type user struct { name string email string } type admin struct { user level string }
嵌入后,被嵌入的類型稱之為內(nèi)部類型,新定義的類型稱之為外部類型。這里user
就是內(nèi)部類型,而admin
是外部類型。
通過嵌入類型,與內(nèi)部類型相關(guān)聯(lián)的所有字段、方法、標(biāo)志符等,都會被外包類型所擁有。就像外部類型自己的一樣,這就達到了代碼快捷復(fù)用組合的目的,而且定義非常簡單,只需聲明這個類型的名字就可以了。
同時,外部類型還可以添加自己的方法、字段屬性等,可以很方便地擴展外部類型的功能。
func main() { ad:=admin{user{"張三","zhangsan@flysnow.org"},"管理員"} fmt.Println("可以直接調(diào)用,名字為:",ad.name) fmt.Println("也可以通過內(nèi)部類型調(diào)用,名字為:",ad.user.name) fmt.Println("但是新增加的屬性只能直接調(diào)用,級別為:",ad.level) }
以上是嵌入類型的使用??梢钥吹?,我們在初始化的時候,采用的是字面值的方式。所以要按其定義的結(jié)構(gòu)進行初始化,先初始化user
這個內(nèi)部類型的,再初始化新增的level
屬性。
對于內(nèi)部類型的屬性和方法訪問,我們可以用外部類型直接訪問,也可以通過內(nèi)部類型進行訪問;但是我們?yōu)橥獠款愋托略龅姆椒▽傩宰侄危荒苁褂猛獠款愋驮L問,因為內(nèi)部類型沒有這些。
當(dāng)然,外部類型也可以聲明同名的字段或者方法,來覆蓋內(nèi)部類型的,這種情況方法比較多,我們以方法為例。
func main() { ad:=admin{user{"張三","zhangsan@flysnow.org"},"管理員"} ad.user.sayHello() ad.sayHello() } type user struct { name string email string } type admin struct { user level string } func (u user) sayHello(){ fmt.Println("Hello,i am a user") } func (a admin) sayHello(){ fmt.Println("Hello,i am a admin") }
內(nèi)部類型user
有一個sayHello
方法,外部類型對其進行了覆蓋,同名重寫sayHello
,然后我們在main方法里分別訪問這兩個類型的方法,打印輸出:
Hello,i am a user Hello,i am a admin
從輸出中看,方法sayHello
被成功覆蓋了。
嵌入類型的強大,還體現(xiàn)在:如果內(nèi)部類型實現(xiàn)了某個接口,那么外部類型也被認(rèn)為實現(xiàn)了這個接口。我們稍微改造下例子看下。
func main() { ad:=admin{user{"張三","zhangsan@flysnow.org"},"管理員"} sayHello(ad.user)//使用user作為參數(shù) sayHello(ad)//使用admin作為參數(shù) } type Hello interface { hello() } func (u user) hello(){ fmt.Println("Hello,i am a user") } func sayHello(h Hello){ h.hello() }
這個例子原來的結(jié)構(gòu)體類型user
和admin
的定義不變,新增了一個接口Hello
,然后讓user
類型實現(xiàn)這個接口,最后我們定義了一個sayHello
方法,它接受一個Hello
接口類型的參數(shù)。最終我們在main函數(shù)演示的時候,發(fā)現(xiàn)不管是user
類型,還是admin
類型作為參數(shù)傳遞給sayHello
方法的時候,都可以正常調(diào)用。
這里就可以說明admin
實現(xiàn)了接口Hello
。但是我們又沒有顯示聲明類型admin
實現(xiàn),所以這個實現(xiàn)是通過內(nèi)部類型user
實現(xiàn)的;因為admin
包含了user
所有的方法函數(shù),所以也就實現(xiàn)了接口Hello
。
當(dāng)然外部類型也可以重新實現(xiàn),只需要像上面例子一樣覆蓋同名的方法即可。這里要說明的是,不管我們?nèi)绾瓮采w,都不會影響內(nèi)部類型,我們還可以通過訪問內(nèi)部類型來訪問它的方法、屬性字段等。
嵌入類型的定義,是Go為了方便我們擴展或者修改已有類型的行為,是為了宣傳組合這個概念而設(shè)計的,所以我們經(jīng)常使用組合,靈活運用組合,擴展出更多的我們需要的類型結(jié)構(gòu)