溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點(diǎn)擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

Go語言接口類型怎么定義

發(fā)布時間:2023-03-29 10:27:54 來源:億速云 閱讀:119 作者:iii 欄目:開發(fā)技術(shù)

今天小編給大家分享一下Go語言接口類型怎么定義的相關(guān)知識點(diǎn),內(nèi)容詳細(xì),邏輯清晰,相信大部分人都還太了解這方面的知識,所以分享這篇文章給大家參考一下,希望大家閱讀完這篇文章后有所收獲,下面我們一起來了解一下吧。

接口

接口是用來定義行為的類型,定義的行為不由接口直接實(shí)現(xiàn),而由通過方法由定義的類型實(shí)現(xiàn)

Golang中,接口是一組方法的簽名,是語言中一個重要的組成部分,其目的是通過引入一個中間層與具體的實(shí)現(xiàn)進(jìn)行分離,達(dá)到解耦合的作用,同時隱藏底層實(shí)現(xiàn),減少關(guān)注點(diǎn)

Golang不同于Java,通過隱式實(shí)現(xiàn)聲明的接口,即只要實(shí)現(xiàn)了接口聲明中的方法,就是實(shí)現(xiàn)了接口,
接口的定義需要使用interface關(guān)鍵字,且在接口中只能定義方法簽名,不能包含成員變量

基于官方的io包進(jìn)行分析:

type Reader interface {
   Read(p []byte) (n int, err error)
}

上面是io包中聲明的Reader接口,如果一個類型需要實(shí)現(xiàn)Reader接口,那么就僅需要實(shí)現(xiàn)Read(p []byte) (n int, err error)方法,如LimitedReader就實(shí)現(xiàn)了Reader接口:

type LimitedReader struct {
    R Reader
    N int64
}

func(l *LimitedReader) Read(p []byte) (n int, err error) {
    if ;.N <= 0 {
        return 0, EOF
    }
    if int64(len(p)) > l.N {
        p = p[o:l.N]
    }
    n, err := l.R.Read(p)
    l.N -= int64(n)
    return
}

Golang只會在參數(shù)傳遞、返回參數(shù)和變量賦值時對類型是否實(shí)現(xiàn)了某個接口進(jìn)行檢查,接口在定義方法時對實(shí)現(xiàn)的接受者做限制,所以會有兩種方式實(shí)現(xiàn)接口:結(jié)構(gòu)體實(shí)現(xiàn)和指針實(shí)現(xiàn)。

但這兩種實(shí)現(xiàn)方式不可以同時存在,Go語言的編譯器會在結(jié)構(gòu)體類型和指針類型都實(shí)現(xiàn)同一個方法時報錯“method redeclared”

type Cat struct {}
type Duck interface {}

func (c Cat) Quack {} // 結(jié)構(gòu)體實(shí)現(xiàn)
func (c *Cat) Quack {} // 指針實(shí)現(xiàn)

var d Duck = Cat{} // 結(jié)構(gòu)體初始化
var d Duck = &Cat{} // 指針初始化

注意:指針實(shí)現(xiàn)接口,結(jié)構(gòu)體初始化變量是無法通過編譯的;而結(jié)構(gòu)體實(shí)現(xiàn)接口,指針初始化變量可以

(Golang在傳遞參數(shù)是值傳遞的,指針初始化變量時,指針可以隱式地獲取到指向的結(jié)構(gòu)體:c.i可以理解成(*c).i)

詳細(xì)理解就是在Golang中,初始化變量后進(jìn)行方法調(diào)用時會發(fā)生`值拷貝`:

1.對于初始化的指針來說,意味著拷貝的新指針仍然與原指針一樣,指向一個相同且唯一的結(jié)構(gòu)體,所以編譯器可以隱式通過對變量的解引用(dereference)獲取到指針的結(jié)構(gòu)體

2.而對于結(jié)構(gòu)體而言,這是拷貝生成了新的結(jié)構(gòu)體,但方法的參數(shù)是指針,編譯器既不可能創(chuàng)建一個新的指針,即使創(chuàng)建也無法指向最初調(diào)用該方法的結(jié)構(gòu)體
具體的例子如Goinaction的代碼示例:

listing36.go

package main

import (
    "fmt"
)

type notifier interface {
    notify()
}

type user struct {
    name string
    email string
}

func (u *user) notify() {
    fmt.Printf("Sending user email to %s<%s>)\n",
    u.name,
    u.email)
}

func main() {
    u := user{"Bill", "bill@email.com"}
    
    sendNotification(u)
}

func sendNotification(n notifier) {
    n.notify()
}

仔細(xì)查看代碼就會發(fā)現(xiàn)u是一個結(jié)構(gòu)體類型,而notify方法是使用指針接受者實(shí)現(xiàn)的,上述代碼自然就無法編譯通過

數(shù)據(jù)結(jié)構(gòu)

Golang根據(jù)接口類型是否包含一組方法將接口類型分成兩類:

1.使用runtime.iface結(jié)構(gòu)體表示包含方法的接口

type iface struct {
      tab *itab // runtime.itab類型結(jié)構(gòu)體,接口類型的核心組成部分
      data unsafe.Pointer // 指向原始數(shù)據(jù)的指針
  }

2.使用runtime.eface結(jié)構(gòu)體表示不包含任何方法的interface{}類型

type eface struct {
      _type *_type // 指向類型的指針
      data unsafe.Pointer // 指向底層數(shù)據(jù)的指針
  }

接口類型不是任意類型

注意:interface{}類型不是任意類型,如果將類型轉(zhuǎn)換成interface{}類型,變量在運(yùn)行期間的類型也會發(fā)生變化,獲取變量類型時會得到interface{}

package main

type Test struct {}

func NilOrNot(v interface{}) bool {

  return v == nil
}

func main() {

  var s *Test
  fmt.Println(s == nil)
  fmt.Println(NilorNot(s))
}

運(yùn)行上述代碼時,第一行打印true,第二行會打印false。

出現(xiàn)上述現(xiàn)象的原因就是在調(diào)用NilOrNot函數(shù)時發(fā)生了隱式的類型轉(zhuǎn)換:*Test類型轉(zhuǎn)換成interface{},除了這種傳參的情況,在變量賦值也是如此

動態(tài)派發(fā)(Dynamic dispatch)是在運(yùn)行期間選擇具體多態(tài)操作(方法或函數(shù))執(zhí)行的過程,接口的引入使得Golang具備了動態(tài)派發(fā)的特性,即在調(diào)用接口類型的方法時,如果編譯期間不能確認(rèn)接口的類型,則會在運(yùn)行期間決定具體調(diào)用該方法的具體實(shí)現(xiàn)

動態(tài)派發(fā)在結(jié)構(gòu)體上的表現(xiàn)非常差,應(yīng)當(dāng)盡量避免使用結(jié)構(gòu)體類型實(shí)現(xiàn)接口

以上就是“Go語言接口類型怎么定義”這篇文章的所有內(nèi)容,感謝各位的閱讀!相信大家閱讀完這篇文章都有很大的收獲,小編每天都會為大家更新不同的知識,如果還想學(xué)習(xí)更多的知識,請關(guān)注億速云行業(yè)資訊頻道。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報,并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI