溫馨提示×

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

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

go語言接口類型如何轉(zhuǎn)換

發(fā)布時(shí)間:2023-01-14 09:56:24 來源:億速云 閱讀:121 作者:iii 欄目:編程語言

這篇文章主要介紹“go語言接口類型如何轉(zhuǎn)換”的相關(guān)知識(shí),小編通過實(shí)際案例向大家展示操作過程,操作方法簡單快捷,實(shí)用性強(qiáng),希望這篇“go語言接口類型如何轉(zhuǎn)換”文章能幫助大家解決問題。

go語言可利用類型斷言來進(jìn)行接口類型。在Go中,無論是將一個(gè)接口類型轉(zhuǎn)換成另一個(gè)接口類型,還是將一個(gè)接口轉(zhuǎn)換為另一個(gè)基本類型,都必須需要使用 類型斷言;轉(zhuǎn)換語法有兩種“轉(zhuǎn)換后的變量 := 接口變量.(目標(biāo)類型)”和“轉(zhuǎn)換后的變量 , ok := 接口變量.(目標(biāo)類型)”。

在 Golang 中,將一個(gè) 接口 類型轉(zhuǎn)換成另一個(gè)接口 類型,或者將一個(gè)接口轉(zhuǎn)換為另一個(gè)基本類型,都必須需要使用 類型斷言

類型斷言的格式

類型斷言是一個(gè)使用在接口值上的操作。語法上它看起來像 i.(T) 被稱為斷言類型,這里 i 表示一個(gè)接口的類型和 T 表示一個(gè)類型。一個(gè)類型斷言檢查它操作對(duì)象的動(dòng)態(tài)類型是否和斷言的類型匹配。

類型斷言的基本格式如下:

t := i.(T)

其中,i 代表接口變量,T 代表轉(zhuǎn)換的目標(biāo)類型,t 代表轉(zhuǎn)換后的變量。

這里有兩種可能。第一種,如果斷言的類型 T 是一個(gè)具體類型,然后類型斷言檢查 i 的動(dòng)態(tài)類型是否和 T 相同。如果這個(gè)檢查成功了,類型斷言的結(jié)果是 i 的動(dòng)態(tài)值,當(dāng)然它的類型是 T。換句話說,具體類型的類型斷言從它的操作對(duì)象中獲得具體的值。如果檢查失敗,接下來這個(gè)操作會(huì)拋出 panic。例如:

var w io.Writer
w = os.Stdout
f := w.(*os.File) // 成功: f == os.Stdout
c := w.(*bytes.Buffer) // 死機(jī):接口保存*os.file,而不是*bytes.buffer

第二種,如果相反斷言的類型 T 是一個(gè)接口類型,然后類型斷言檢查是否 i 的動(dòng)態(tài)類型滿足 T。如果這個(gè)檢查成功了,動(dòng)態(tài)值沒有獲取到;這個(gè)結(jié)果仍然是一個(gè)有相同類型和值部分的接口值,但是結(jié)果有類型 T。換句話說,對(duì)一個(gè)接口類型的類型斷言改變了類型的表述方式,改變了可以獲取的方法集合(通常更大),但是它保護(hù)了接口值內(nèi)部的動(dòng)態(tài)類型和值的部分。

在下面的第一個(gè)類型斷言后,w 和 rw 都持有 os.Stdout 因此它們每個(gè)有一個(gè)動(dòng)態(tài)類型 *os.File,但是變量 w 是一個(gè) io.Writer 類型只對(duì)外公開出文件的 Write 方法,然而 rw 變量也只公開它的 Read 方法。

var w io.Writer
w = os.Stdout
rw := w.(io.ReadWriter) // 成功:*os.file具有讀寫功能
w = new(ByteCounter)
rw = w.(io.ReadWriter) // 死機(jī):*字節(jié)計(jì)數(shù)器沒有讀取方法

如果斷言操作的對(duì)象是一個(gè) nil 接口值,那么不論被斷言的類型是什么這個(gè)類型斷言都會(huì)失敗。幾乎不需要對(duì)一個(gè)更少限制性的接口類型(更少的方法集合)做斷言,因?yàn)樗憩F(xiàn)的就像賦值操作一樣,除了對(duì)于 nil 接口值的情況。

如果 i 沒有完全實(shí)現(xiàn) T 接口的方法,這個(gè)語句將會(huì)觸發(fā)宕機(jī)。觸發(fā)宕機(jī)不是很友好,因此上面的語句還有一種寫法:

t,ok := i.(T)

這種寫法下,如果發(fā)生接口未實(shí)現(xiàn)時(shí),將會(huì)把 ok 置為 false,t 置為 T 類型的 0 值。正常實(shí)現(xiàn)時(shí),ok 為 true。這里 ok 可以被認(rèn)為是:i 接口是否實(shí)現(xiàn) T 類型的結(jié)果。

將接口轉(zhuǎn)換為其他接口

實(shí)現(xiàn)某個(gè)接口的類型同時(shí)實(shí)現(xiàn)了另外一個(gè)接口,此時(shí)可以在兩個(gè)接口間轉(zhuǎn)換。

鳥和豬具有不同的特性,鳥可以飛,豬不能飛,但兩種動(dòng)物都可以行走。如果使用結(jié)構(gòu)體實(shí)現(xiàn)鳥和豬,讓它們具備自己特性的 Fly() 和 Walk() 方法就讓鳥和豬各自實(shí)現(xiàn)了飛行動(dòng)物接口(Flyer)和行走動(dòng)物接口(Walker)。

將鳥和豬的實(shí)例創(chuàng)建后,被保存到 interface{} 類型的 map 中。interface{} 類型表示空接口,意思就是這種接口可以保存為任意類型。對(duì)保存有鳥或豬的實(shí)例的 interface{} 變量進(jìn)行斷言操作,如果斷言對(duì)象是斷言指定的類型,則返回轉(zhuǎn)換為斷言對(duì)象類型的接口;如果不是指定的斷言類型時(shí),斷言的第二個(gè)參數(shù)將返回 false。

例如下面的代碼:

var obj interface = new(bird)
f, isFlyer := obj.(Flyer)

代碼中,new(bird) 產(chǎn)生 *bird 類型的 bird 實(shí)例,這個(gè)實(shí)例被保存在 interface{} 類型的 obj 變量中。使用 obj.(Flyer) 類型斷言,將 obj 轉(zhuǎn)換為 Flyer 接口。f 為轉(zhuǎn)換成功時(shí)的 Flyer 接口類型,isFlyer 表示是否轉(zhuǎn)換成功,類型就是 bool。

下面是詳細(xì)的代碼(代碼1):

package main
import "fmt"
// 定義飛行動(dòng)物接口
type Flyer interface {
    Fly()
}
// 定義行走動(dòng)物接口
type Walker interface {
    Walk()
}
// 定義鳥類
type bird struct {
}
// 實(shí)現(xiàn)飛行動(dòng)物接口
func (b *bird) Fly() {
    fmt.Println("bird: fly")
}
// 為鳥添加Walk()方法, 實(shí)現(xiàn)行走動(dòng)物接口
func (b *bird) Walk() {
    fmt.Println("bird: walk")
}
// 定義豬
type pig struct {
}
// 為豬添加Walk()方法, 實(shí)現(xiàn)行走動(dòng)物接口
func (p *pig) Walk() {
    fmt.Println("pig: walk")
}
func main() {
// 創(chuàng)建動(dòng)物的名字到實(shí)例的映射
    animals := map[string]interface{}{
        "bird": new(bird),
        "pig":  new(pig),
    }
    // 遍歷映射
    for name, obj := range animals {
        // 判斷對(duì)象是否為飛行動(dòng)物
        f, isFlyer := obj.(Flyer)
        // 判斷對(duì)象是否為行走動(dòng)物
        w, isWalker := obj.(Walker)
        fmt.Printf("name: %s isFlyer: %v isWalker: %v\n", name, isFlyer, isWalker)
        // 如果是飛行動(dòng)物則調(diào)用飛行動(dòng)物接口
        if isFlyer {
            f.Fly()
        }
        // 如果是行走動(dòng)物則調(diào)用行走動(dòng)物接口
        if isWalker {
            w.Walk()
        }
    }
}

代碼說明如下:

  • 第 6 行定義了飛行動(dòng)物的接口。

  • 第 11 行定義了行走動(dòng)物的接口。

  • 第 16 和 30 行分別定義了鳥和豬兩個(gè)對(duì)象,并分別實(shí)現(xiàn)了飛行動(dòng)物和行走動(dòng)物接口。

  • 第 41 行是一個(gè) map,映射對(duì)象名字和對(duì)象實(shí)例,實(shí)例是鳥和豬。

  • 第 47 行開始遍歷 map,obj 為 interface{} 接口類型。

  • 第 50 行中,使用類型斷言獲得 f,類型為 Flyer 及 isFlyer 的斷言成功的判定。

  • 第 52 行中,使用類型斷言獲得 w,類型為 Walker 及 isWalker 的斷言成功的判定。

  • 第 57 和 62 行,根據(jù)飛行動(dòng)物和行走動(dòng)物兩者是否斷言成功,調(diào)用其接口。

代碼輸出如下:

go語言接口類型如何轉(zhuǎn)換

將接口轉(zhuǎn)換為其他類型

在代碼 1 中,可以實(shí)現(xiàn)將接口轉(zhuǎn)換為普通的指針類型。例如將 Walker 接口轉(zhuǎn)換為 *pig 類型,請(qǐng)參考下面的代碼:

p1 := new(pig)
var a Walker = p1
p2 := a.(*pig)
fmt.Printf("p1=%p p2=%p", p1, p2)

對(duì)代碼的說明如下:

  • 第 3 行,由于 pig 實(shí)現(xiàn)了 Walker 接口,因此可以被隱式轉(zhuǎn)換為 Walker 接口類型保存于 a 中。

  • 第 4 行,由于 a 中保存的本來就是 *pig 本體,因此可以轉(zhuǎn)換為 *pig 類型。

  • 第 6 行,對(duì)比發(fā)現(xiàn),p1 和 p2 指針是相同的。

如果嘗試將上面這段代碼中的 Walker 類型的 a 轉(zhuǎn)換為 *bird 類型,將會(huì)發(fā)出運(yùn)行時(shí)錯(cuò)誤,請(qǐng)參考下面的代碼:

p1 := new(pig)
var a Walker = p1
p2 := a.(*bird)

運(yùn)行時(shí)報(bào)錯(cuò):

panic: interface conversion: main.Walker is *main.pig, not *main.bird

報(bào)錯(cuò)意思是:接口轉(zhuǎn)換時(shí),main.Walker 接口的內(nèi)部保存的是 *main.pig,而不是 *main.bird。

因此,接口在轉(zhuǎn)換為其他類型時(shí),接口內(nèi)保存的實(shí)例對(duì)應(yīng)的類型指針,必須是要轉(zhuǎn)換的對(duì)應(yīng)的類型指針。

關(guān)于“go語言接口類型如何轉(zhuǎn)換”的內(nèi)容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業(yè)相關(guān)的知識(shí),可以關(guān)注億速云行業(yè)資訊頻道,小編每天都會(huì)為大家更新不同的知識(shí)點(diǎn)。

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

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

AI