這個(gè) Go 語言的經(jīng)典 “坑”,我算是服了
大家好,我是明哥。
在開始之前,先考你一個(gè)非常 Go 味的經(jīng)典問題:如何判斷一個(gè) interface{} 的值是否為 nil ?
這也是面試有可能會(huì)被問到的一個(gè)問題,這個(gè)問題很 “迷”,平時(shí)沒有特別留心的朋友,很容易在這邊裁了。
我相信很多人會(huì)下意識(shí)的回答,直接 v == nil 進(jìn)行判斷不就好了嗎?
很久之前,我也是那么想的,可寫了個(gè) demo 后,才發(fā)現(xiàn)事情沒那么簡(jiǎn)單。
請(qǐng)看下面這段代碼,可以先猜測(cè)一下輸出的結(jié)果。
- package main
- import (
- "fmt"
- )
- func main() {
- var a *string = nil
- var b interface{} = a
- fmt.Println(b==nil)
- }
答案應(yīng)該會(huì)跟你下意識(shí)的回答 相反。
輸出的結(jié)果的是 false
這是為什么呢?接下來,我們就要好好的嘮嘮這里面的道道。
1. 兩個(gè) interface 比較
interface 的內(nèi)部實(shí)現(xiàn)包含了兩個(gè)字段,一個(gè)是 type,一個(gè)是 data
對(duì)于這樣一個(gè)變量
- var age interface{} = 25
其實(shí)內(nèi)部結(jié)構(gòu)是
因此兩個(gè) interface 比較,勢(shì)必與這兩個(gè)字段有所關(guān)系。
經(jīng)過驗(yàn)證,只有下面兩種情況,兩個(gè) interface 才會(huì)相等。
第一種情況
type 和 data 都相等
在下面的代碼中,p1 和 p2 的 type 都是 Profile,data 都是 {"iswbm"},因此 p1 與 p2 相等
而 p3 和 p3 雖然類型都是 *Profile,但由于 data 存儲(chǔ)的是結(jié)構(gòu)體的地址,而兩個(gè)地址和不相同,因此 p3 與 p4 不相等
- package main
- import "fmt"
- type Profile struct {
- Name string
- }
- type ProfileInt interface {}
- func main() {
- var p1, p2 ProfileInt = Profile{"iswbm"}, Profile{"iswbm"}
- var p3, p4 ProfileInt = &Profile{"iswbm"}, &Profile{"iswbm"}
- fmt.Printf("p1 --> type: %T, data: %v \n", p1, p1)
- fmt.Printf("p2 --> type: %T, data: %v \n", p2, p2)
- fmt.Println(p1 == p2) // true
- fmt.Printf("p3 --> type: %T, data: %p \n", p3, p3)
- fmt.Printf("p4 --> type: %T, data: %p \n", p4, p4)
- fmt.Println(p3 == p4) // false
- }
運(yùn)行后,輸出如下
- p1 --> type: main.Profile, data: {iswbm}
- p2 --> type: main.Profile, data: {iswbm}
- true
- p3 --> type: *main.Profile, data: 0xc00008e200
- p4 --> type: *main.Profile, data: 0xc00008e210
- false
第二種情況
特殊情況:兩個(gè) interface 都是 nil
當(dāng)一個(gè) interface 的 type 和 data 都處于 unset 狀態(tài)的時(shí)候,那么該 interface 的類型和值都為 nil
- package main
- import "fmt"
- func main() {
- var p1, p2 interface{}
- fmt.Println(p1 == p2) // true
- fmt.Println(p1 == nil) // true
2. interface 與 非 interface 比較
當(dāng) interface 與非 interface 比較時(shí),會(huì)將 非interface 轉(zhuǎn)換成 interface ,然后再按照 兩個(gè) interface 比較 的規(guī)則進(jìn)行比較。
示例如下
- package main
- import (
- "fmt"
- "reflect"
- )
- func main() {
- var a string = "iswbm"
- var b interface{} = "iswbm"
- fmt.Println(a==b) // true
- }
上面這種例子可能還好理解,那么請(qǐng)你看下面這個(gè)例子(文章開頭的示例),為什么 b 與 nil 不相等?
- package main
- import (
- "fmt"
- )
- func main() {
- var a *string = nil
- var b interface{} = a
- fmt.Println(b==nil) // false
- }
但當(dāng)你使用 b==nil 進(jìn)行判斷時(shí),其實(shí)右邊的 nil 并非單純的是我們所理解的值為nil,而正確的理解應(yīng)該是 類型為 nil 且 值也為 nil。
Go 會(huì)先將 nil 轉(zhuǎn)換為interface (type=nil, data=nil) ,這與 b (type=*string, data=nil) 雖然 data 是一樣的,但 type 不相等,因此他們并不相等。
那有沒有辦法判斷一個(gè) interface{} 是不是 nil 呢?
有辦法的,但是要借助反射,一個(gè)非萬不得已不會(huì)使用的 reflect 包。
- package main
- import (
- "fmt"
- "reflect"
- )
- func IsNil(i interface{}) bool {
- vi := reflect.ValueOf(i)
- if vi.Kind() == reflect.Ptr {
- return vi.IsNil()
- }
- return false
- }
- func main() {
- var a *string = nil
- var b interface{} = a
- fmt.Println(IsNil(b))
- }
本文通過一些例子介紹了 Go 在比較時(shí)候,內(nèi)部的一些實(shí)現(xiàn)原理,對(duì)于很多人,可能是一個(gè)“新”知識(shí),沒有掌握的話,一定會(huì)在以后在編碼過程中給自己挖了個(gè)“大坑”,而這種語言內(nèi)部的 “坑”,不知道就是不知道,再怎么代碼走查都很難發(fā)現(xiàn)。希望通過本篇文章,帶你一起把這個(gè)“坑” 給填上。
本篇原屬于 Go 面試題庫專欄系列文章,以前都是在標(biāo)題上寫明了是面試題,考慮到有些人最近沒有在面試,怕你們錯(cuò)過這類即使不面試,也要掌握的知識(shí),以后的內(nèi)容,可能不會(huì)在標(biāo)題上特別標(biāo)明是面試題了。
本專欄系列文章,我都公開到 Github 上:https://github.com/iswbm/golang-interview
這個(gè)號(hào)沒有留言功能呢 ,如果文章有寫得不對(duì)的地方,可以去那里提交 issue 幫我指正。順便可以幫我點(diǎn)個(gè)小 ??,在那里我對(duì)題庫進(jìn)行了分類整理,方便索引查找。
加油噢,我們下篇見!