面試官:值為 nil 為什么不等于 nil ?
var?f?func()
var?a?*struct{}
list?:=?[]interface{}{f,?a}
for?_,?item?:=?range?list?{
?if?item?==?nil?{
??fmt.Println("nil")
?}
}
請(qǐng)問輸出結(jié)果是啥?
可能很多同學(xué),會(huì)認(rèn)為結(jié)果是輸出兩個(gè) nil。
為什么呢?
因?yàn)?f 和 a 都沒初始化,都是 nil,所以循環(huán)遍歷后肯定也是 nil。
如果你的答案也是這樣,那就掉進(jìn)坑里了。
答案是,啥都不會(huì)輸出!
且聽我慢慢給你分析:
一、變量的值和類型
我們先來打印下這兩個(gè)值:
var?f?func()
var?a?*struct{}
fmt.Println(f,?a)
//?輸出結(jié)果
<nil>?<nil>
我們這樣打印其實(shí)是打印的是他的值,是 nil 沒錯(cuò)的。
但是類型不是 nil。
我們可以這樣打印他類型:
var?f?func()
var?a?*struct{}
fmt.Printf("%T,%T?\n",?f,?a)
//?輸出結(jié)果
func(),*struct?{}
二、if 判 nil 含有對(duì)類型的判斷
當(dāng)我們從 interface ?里面把對(duì)象取出來后,使用 if 進(jìn)行判斷,他不單單的比較的是值,還有類型。
看下這段代碼:
var?f?func()
var?a?*struct{}
list?:=?[]interface{}{f,?a,?nil}
?for?_,?item?:=?range?list?{
??fmt.Println("item=",?item)
??fmt.Printf("item?type:?%T?\n",?item)
??if?item?==?nil?{
???fmt.Println("item?==?nil")
??}
??fmt.Println("----")
?}
}
現(xiàn)在看下運(yùn)行結(jié)果:
$?go?run?main.go?
item=?<nil>
item?type:?func()?
----
item=?<nil>
item?type:?*struct?{}?
----
item=?<nil>
item?type:?<nil>?
item?==?nil
----
你會(huì)發(fā)現(xiàn)之后最后 nil 的判斷是通過的,前面兩個(gè)判斷都不通過。
三、怎么判斷值是否為 nil
我們?cè)趯懘a時(shí),最好是盡量避免這種代碼,如果硬要這么寫,那我們可以通過以下兩種常見方式判 nil。
1、斷言
list?:=?[]interface{}{f,?a}
for?_,?item?:=?range?list?{
?if?v,?ok?:=?item.(func());?ok?&&?v?==?nil?{
??fmt.Println("item?is?nil")
?}
?if?v,?ok?:=?item.(*struct{});?ok?&&?v?==?nil?{
??fmt.Println("item?is?nil")
?}
}
2、反射
list?:=?[]interface{}{f,?a}
for?_,?item?:=?range?list?{
?if?reflect.ValueOf(item).IsNil()?{
??fmt.Println("item?is?nil")
?}
}
你學(xué)廢了么?
原文:https://mp.weixin.qq.com/s/qPrfMgPVkl1gSLxR690zgw