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。

推薦閱讀: