Go語言之嵌入類型詳解
一、什麼是嵌入類型
先看如下代碼:
type user struct { name string email string } type admin struct { user // Embedded Type level string }
可以看到admin結構中的一個成員是user,那麼admin中就嵌入瞭user類型。
- admin也叫做外部類型
- user也叫做內部類型
二、外部類型和內部類型之間的關系和機制
func (u *user) notify() { fmt.Printf("Sending user email to %s<%s>\n", u.name, u.email) }
如上代碼,實現瞭一個方法notify(),接收者是 *user。
func main() { // Create an admin user. ad := admin{ user: user{ name: "john smith", email: "[email protected]", }, level: "super", } // We can access the inner type's method directly. ad.user.notify() // The inner type's method is promoted. ad.notify() }
main函數中定義瞭一個變量ad,並且進行瞭賦值
運行:
Sending user email to john smith<[email protected]>
Sending user email to john smith<[email protected]>
①沒有編譯錯誤
②notify()可以被ad.user調用是可以理解的,但是ad.notify()也能執行是為什麼。
這裡涉及到瞭一個嵌入類型背後的機制,內部類型提升 (感覺有點像C#、Java裡面的繼承,user是父類,admin是子類,admin的實例對象直接調用瞭父類的notify方法。)
進一步研究:我們再定義一個接口、以及一個接受該接口的函數。
接口,隻有一個方法notify
type notifier interface { notify() }
函數,接受一個實現notifier接口的類型實例,內部就是調用notify方法
func sendNotification(n notifier) { n.notify() }
main方法如下
func main() { // Create an admin user. ad := admin{ user: user{ name: "john smith", email: "[email protected]", }, level: "super", } var user = ad.user sendNotification(&user) sendNotification(&ad) }
運行結果:
Sending user email to john smith<[email protected]>
Sending user email to john smith<[email protected]>
①可以看到這裡傳入 &user和&ad都是可以的,說明類型提升導致admin也是實現瞭notifier接口瞭。
②為什麼穿&user和&ad,而不是直接傳user和ad,這就涉及到瞭之前總結過的【方法集】的概念瞭。復習一下:
從上面兩個表,可以知道由於方法的接收者是 *user ,所以說隻有*user實現瞭該接口的方法,這就是為什麼輸入&user、&ad瞭
再進一步研究:我們在C#當中,如果使用瞭virtual作為修飾符在父類中寫瞭一個方法,那麼在子類中通過override可以重寫這個方法,最終的結果就是調用的非父類的該方法,而是子類的,Go語言同樣可以。
例如
// 通過admin 類型值的指針 // 調用的方法 func (a *admin) notify() { fmt.Printf("Sending admin email to %s<%s>\n", a.name, a.email) }
在剛剛的代碼中,追加一個*admin作為接受者的方法。
運行結果如下:
Sending user email to john smith<[email protected]>
Sending admin email to john smith<[email protected]>
可以發現此時這兩此運行的結果就不一樣瞭,第二次sendNotification(&ad)調用的notify方法就是admin這個類型的瞭。
這表明,如果外部類型實現瞭notify 方法,內部類型的實現就不會被提升。不過內部類型的值一直存在,因此還可以通過直接訪問內部類型的值,來調用沒有被提升的內部類型實現的方法。
三、總結
綜上:嵌入類型為Go語言類型提供瞭一種很好的擴展能力,通過內部類型的提升,使得外部類型擁有瞭內部類型的方法,也可以通過外部類型實現同樣的方法來替代內部類型的。總體來說很像C#語言中的繼承。
到此這篇關於Go語言之嵌入類型的文章就介紹到這瞭。希望對大傢的學習有所幫助,也希望大傢多多支持WalkonNet。