本文準(zhǔn)備介紹interface
的一些其他方法。關(guān)于interface
的基礎(chǔ)知識(shí)可以看一下另一篇文章:Go語(yǔ)言多態(tài)和interface的使用
萬(wàn)能類(lèi)型interface
在Java
以及其他語(yǔ)言當(dāng)中接口是一種寫(xiě)法規(guī)范,而在golang
當(dāng)中,interface
其實(shí)也是一種值,它可以像是值一樣傳遞。并且在它的底層,它其實(shí)是一個(gè)值和類(lèi)型的元組。
這里我們來(lái)看下golang
官方文檔當(dāng)中的一個(gè)例子:
package main
import (
"fmt"
"math"
)
type I interface {
M()
}
type T struct {
S string
}
func (t *T) M() {
fmt.Println(t.S)
}
type F float64
func (f F) M() {
fmt.Println(f)
}
func main() {
var i I
i = &T{"Hello"}
describe(i)
i.M()
i = F(math.Pi)
describe(i)
i.M()
}
func describe(i I) {
fmt.Printf("(%v, %T)\n", i, i)
}
在上面的代碼當(dāng)中定義了一個(gè)叫做describe的方法,在這個(gè)方法當(dāng)中我們輸出了兩個(gè)值,一個(gè)是接口i對(duì)應(yīng)的值,另一個(gè)是接口i的類(lèi)型。
(推薦課程:Go教程)
我們輸出的結(jié)果如下:
可以看到接口當(dāng)中既存儲(chǔ)了對(duì)應(yīng)的結(jié)構(gòu)體的實(shí)例的信息,也存儲(chǔ)了結(jié)構(gòu)體的類(lèi)型。因此interface
可以理解成一種特殊的類(lèi)型。
實(shí)際上也的確如此,我們可以把interface
理解成一種萬(wàn)能數(shù)據(jù)類(lèi)型,它可以接收任何類(lèi)型的值。我們看下下面這種用法:
var a1 interface{} = 1
var a2 interface{} = "abc"
list := make([]interface{}, 0)
list = append(list, a1)
list = append(list, a2)
fmt.Println(list)
在代碼當(dāng)中我們創(chuàng)建了一個(gè)interface{}
類(lèi)型的slice
,它可以接收任何類(lèi)型的值和實(shí)例。另外我們用interface{}
這個(gè)類(lèi)型也可以接收任何結(jié)構(gòu)體的值。這里可能會(huì)有些迷惑,其實(shí)很容易想明白。interface
表示一種類(lèi)型,可以接收任何實(shí)現(xiàn)了interface
當(dāng)中規(guī)定的方法的類(lèi)型的值。當(dāng)我們定義inteface{}
的時(shí)候,其實(shí)是定義了空的interface
,相當(dāng)于不需要實(shí)現(xiàn)任何方法的空interface
,所以任何類(lèi)型都可以接收,這也就是它成為萬(wàn)能類(lèi)型的原因。
我們接收當(dāng)然沒(méi)有問(wèn)題,問(wèn)題是我們?cè)趺词褂眠@些interface
類(lèi)型的值呢?
一種方法是我們可以判斷一個(gè)interface
的變量類(lèi)型。判斷的方法非常簡(jiǎn)單,我們?cè)?code>interface的變量后面用.(type)
的方法來(lái)判斷。它和map
的key
值判斷一樣,會(huì)返回一個(gè)值和bool
類(lèi)型的標(biāo)記。我們可以通過(guò)這個(gè)標(biāo)記判斷這個(gè)類(lèi)型是否正確。
if v, ok := a1.(int); ok {
fmt.Println(v)
}
如果類(lèi)型比較多的話使用switch
也是可以的:
switch v := i.(type) {
case int:
fmt.Println("int")
case string:
fmt.Println("string")
}
空值nil
interface
類(lèi)型的空值是nil
,和Python
當(dāng)中的None
是一個(gè)意思,表示一個(gè)指針指向空。如果我們?cè)?code>Java或者是其他語(yǔ)言當(dāng)中對(duì)一個(gè)空指針調(diào)用方法,那么會(huì)觸發(fā)NullPointerMethodError
,也就是空指針報(bào)錯(cuò)。這也是我們初學(xué)者在編程當(dāng)中最容易遇到的錯(cuò)誤,往往原因是忘記了對(duì)聲明進(jìn)行初始化導(dǎo)致的。
但是在golang
當(dāng)中不會(huì),即使是nil
也可以調(diào)用interface
的方法。舉個(gè)例子:
type T struct {
S string
}
func (t *T) M() {
fmt.Println(t.S)
}
func main() {
var i I
var t *T
i = t
i.M()
}
我們將t
賦值給了i
,問(wèn)題是t
并沒(méi)有進(jìn)行初始化,所以它是一個(gè)nil
,那么我們的i
也就會(huì)是一個(gè)nil
。我們對(duì)nil
調(diào)用M
方法,在M
方法當(dāng)中我們打印了t
的局部變量S
。由于t
此刻是一個(gè)nil
,它并沒(méi)有這個(gè)變量,所以會(huì)引發(fā)一個(gè)invalid memory address or nil pointer derefernce
的錯(cuò)誤,也就是對(duì)空指針進(jìn)行尋址的錯(cuò)誤。
要解決這個(gè)錯(cuò)誤,其實(shí)很簡(jiǎn)單,我們可以在M
方法當(dāng)中對(duì)t
進(jìn)行判斷,如果發(fā)現(xiàn)t
是一個(gè)nil
,那么我們則跳過(guò)執(zhí)行的邏輯。當(dāng)我們把M
函數(shù)改成這樣之后,就不會(huì)觸發(fā)空指針的問(wèn)題了。
func (t *T) M() {
if t == nil {
fmt.Println("nil")
return
}
fmt.Println(t.S)
}
nil
觸發(fā)異常的問(wèn)題也是初學(xué)者經(jīng)常遇到的問(wèn)題之一,這也要求我們?cè)趯?shí)現(xiàn)結(jié)構(gòu)體內(nèi)方法的時(shí)候一定要記得判斷調(diào)用的對(duì)象是否為nil
,避免不必要的問(wèn)題。
(推薦課程:Go Web編程)
賦值的類(lèi)型選擇
我們都知道golang
當(dāng)中通過(guò)interface
來(lái)實(shí)現(xiàn)多態(tài),只要是實(shí)現(xiàn)了interface
當(dāng)中定義的函數(shù),那么我們就可以將對(duì)應(yīng)的實(shí)例賦值給這個(gè)interface
類(lèi)型。
這看起來(lái)沒(méi)有問(wèn)題,但是在實(shí)際執(zhí)行的時(shí)候仍然會(huì)有一點(diǎn)點(diǎn)小小的問(wèn)題。比如說(shuō)我們有這樣一段代碼:
type Integer int
type Operation interface {
Less(b Integer) bool
Add(b Integer)
}
func (a Integer) Less(b Integer) bool {
return a < b
}
func (a *Integer) Add(b Integer) {
*a += b
}
這段代碼非常簡(jiǎn)單,我們定義了一個(gè)Operation
的interface
,并且實(shí)現(xiàn)了Integer
類(lèi)型的兩個(gè)方法。表面上看一切正常,但是有一個(gè)細(xì)節(jié)。Less
和Add
這兩個(gè)方法針對(duì)的類(lèi)型是不同的,Less
方法我們不需要修改原值,所以我們傳入的是Integer
的值,而Add
方法,我們需要修改原值, 所以我們傳入的類(lèi)型是Integer
的指針。
那么問(wèn)題來(lái)了,這兩個(gè)方法的類(lèi)型不同, 我們還可以將它的值賦值給Operation
這個(gè)interface
嗎?如果可以的話,我們應(yīng)該傳遞的是值還是指針呢?下面代碼當(dāng)中的第二行和第三行究竟哪個(gè)是正確的呢?
var a Integer = 1
var b Operation = &a
var b Operation = a
答案是第二行的是正確的,原因也很簡(jiǎn)單,因?yàn)槲覀儌魅胫羔樦螅?code>golang的編譯器會(huì)自動(dòng)生成一個(gè)新的Less
方法。在這個(gè)轉(zhuǎn)換了類(lèi)型的方法當(dāng)中去調(diào)用了原本的方法,相當(dāng)于做了一層中轉(zhuǎn)。
func (a *Integer) Less(b Integer) bool{
return (*a).Less(b)
}
那反過(guò)來(lái)行不行呢?我們也寫(xiě)出代碼:
func (a Integer) Add (b Integer) {
(&a).Add(b)
}
顯然這樣是不行的,因?yàn)楹瘮?shù)執(zhí)行之后修改的只能是Add
這個(gè)方法當(dāng)中a
這個(gè)參數(shù)的值,而沒(méi)辦法修改原值。這和我們想要的不符合,所以golang
沒(méi)有選擇這種策略。
(推薦微課:Go微課)
總結(jié)
在今天的文章當(dāng)中我們介紹了golang
當(dāng)中interface
的一些高級(jí)用法,比如將它作為萬(wàn)能類(lèi)型來(lái)接收各種格式的值。比如interface
的空指針調(diào)用問(wèn)題,以及interface
中的兩個(gè)函數(shù)接收類(lèi)型不一致的問(wèn)題。
也就是說(shuō)在go
語(yǔ)言當(dāng)中,interface
既是一種多態(tài)實(shí)現(xiàn)的規(guī)范,又有全能類(lèi)型這樣衍生的功能,這個(gè)設(shè)計(jì)的確是很驚艷的。對(duì)interface
的熟練使用可以在一些問(wèn)題當(dāng)中大大降低我們編碼的復(fù)雜度,以及運(yùn)行的效率。這也是golang
的原生優(yōu)勢(shì)之一。希望以上的相關(guān)介紹能對(duì)大家有所幫助。
文章參考來(lái)源:www.toutiao.com/a6859567247216771587/