在 Golang 里, _
(下劃線(xiàn))是個(gè)特殊的標(biāo)識(shí)符。前幾天看 gin 源碼,看到一個(gè)有意思的用法。雖然網(wǎng)上的總結(jié)博客已有很多,但是總是有點(diǎn)欠缺,于是就有了這一篇,方便以后查閱。
用在 import
在導(dǎo)包的時(shí)候,常見(jiàn)這個(gè)用法,尤其是項(xiàng)目中使用到 mysql 或者使用 pprof 做性能分析時(shí),比如
import _ "net/http/pprof"
import _ "github.com/go-sql-driver/mysql"
這種用法,會(huì)調(diào)用包中的init()
函數(shù),讓導(dǎo)入的包做初始化,但是卻不使用包中其他功能。
用在返回值
該用法也是一個(gè)常見(jiàn)用法。Golang 中的函數(shù)返回值一般是多個(gè),err 通常在返回值最后一個(gè)值。但是,有時(shí)候函數(shù)返回值中的某個(gè)值我們不關(guān)心,如何接收了這個(gè)值但不使用,代碼編譯會(huì)報(bào)錯(cuò),因此需要將其忽略掉。比如
for _, val := range Slice {}
_, err := func()
用在變量
我們都知道 Go 語(yǔ)言的接口是非侵入式的,不像 java 和 c++ 那么重,一個(gè)結(jié)構(gòu)體只要實(shí)現(xiàn)了接口定義的所有函數(shù),我們就說(shuō)這個(gè)接口實(shí)現(xiàn)了該接口。有個(gè)專(zhuān)門(mén)的名字表示這種行為,duck typing,即當(dāng)看到一只鳥(niǎo)走起來(lái)像鴨子、游泳起來(lái)像鴨子、叫起來(lái)也像鴨子,那么這只鳥(niǎo)就可以被稱(chēng)為鴨子。
type I interface {
Sing()
}
type T struct {
}
func (t T) Sing() {
}
type T2 struct {
}
func (t *T2) Sing() {
}
// 編譯通過(guò)
var _ I = T{}
// 編譯通過(guò)
var _ I = T{}
// 編譯失敗
var _ I = T2{}
// 編譯通過(guò)
var _ I = T2{}
在這里下劃線(xiàn)用來(lái)判斷結(jié)構(gòu)體是否實(shí)現(xiàn)了接口,如果沒(méi)有實(shí)現(xiàn),在編譯的時(shí)候就能暴露出問(wèn)題,如果沒(méi)有這個(gè)判斷,后代碼中使用結(jié)構(gòu)體沒(méi)有實(shí)現(xiàn)的接口方法,在編譯器是不會(huì)報(bào)錯(cuò)的。
可以看到上面四個(gè)判斷只有第三個(gè)編譯時(shí)失敗的,報(bào)錯(cuò)如下:
./test.go:27:5: cannot use T2 literal (type T2) as type I in assignment:
T2 does not implement I (Sing method has pointer receiver)
這是為什么呢?仔細(xì)看上面代碼發(fā)現(xiàn),T
實(shí)現(xiàn)了 Sing
方法, *T2
實(shí)現(xiàn)了 Sing
方法。
我們都知道,Go 語(yǔ)言中是按值傳遞的。
那對(duì)于 T2
來(lái)說(shuō),調(diào)用 Sing
方法時(shí),copy 一個(gè)副本,然后取地址,通過(guò)這個(gè)地址是找不到原始調(diào)用的那個(gè)結(jié)構(gòu)體的,但是 receiver 是個(gè)指針,表示此次調(diào)用是需要改變調(diào)用者內(nèi)部變量的,很明顯,以 T2
類(lèi)型調(diào)用無(wú)法完達(dá)到這個(gè)目的,所以這里是需要報(bào)錯(cuò)的。而以 T2
調(diào)用 Sing
方法,則可以,因此不報(bào)錯(cuò)。
而對(duì)于 T
來(lái)說(shuō),不管是否有指針調(diào)用,都不會(huì)報(bào)錯(cuò),實(shí)際上,Go 語(yǔ)言會(huì)自動(dòng)實(shí)現(xiàn) *T
的 Sing
方法。
當(dāng)然,這些都是我的個(gè)人理解,如果不對(duì)的話(huà),歡迎斧正。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。