go語(yǔ)言?nil使用避坑指南
引言
今天筆試題遇到 var x string = nil
,問(wèn)這個(gè)定義是否正確?
這里給出答案:
cannot use nil as string value in variable declaration。
也就是說(shuō),string
類型和nil
八竿子打不著,要想判斷字符串是否為空,可以使用str == ""
或者len(str) == 0
。
接下來(lái),順便總結(jié)一下nil的使用
nil
nil
是go語(yǔ)言中預(yù)先定義的標(biāo)識(shí)符,不是關(guān)鍵字或保留字。 我們可以直接使用nil
,而不用聲明它。 而且我們可以定義一個(gè)名稱為 nil
的變量,比如下面這樣:
var nil = errors.New("nil") fmt.Printf("%#v\n", nil)//&errors.errorString{s:"nil"}
雖然上面的聲明語(yǔ)句可以通過(guò)編譯,但是并不提倡這么做。
默認(rèn)值nil (重點(diǎn)記住)
在go語(yǔ)言中:
- 布爾類型的零值(初始值)為
false
- 數(shù)值類型的零值為
0
- 字符串類型的零值為空字符串
""
除此之外其它類型的默認(rèn)值為nil
,nil
可以代表下面這些類型的零值:
- 指針類型(包括
unsafe
中的) map
類型slice
類型function
類型channel
類型interface
類型
nil沒(méi)有默認(rèn)類型
預(yù)先定義的nil
是唯一的一個(gè)go語(yǔ)言中沒(méi)有默認(rèn)類型的非類型值。對(duì)于編譯器來(lái)說(shuō),必須從上下文中獲取充足的信息才能推斷出nil
的類型。
當(dāng)你把nil
賦值給一個(gè)channel
類型變量,此時(shí)為channel
類型。
當(dāng)你把nil
賦值給map
類型變量,此時(shí)為map
類型。
不同類型的nil值占用的內(nèi)存大小可能是不一樣的
一個(gè)類型的所有的值的內(nèi)存布局都是一樣的。nil也不例外。nil的大小一致與同類型中的非nil類型的值的大小一樣大。但是不同類型的nil值的大小可能不同.
package main import ( "fmt" "unsafe" ) func main() { var p *struct{} = nil fmt.Println(unsafe.Sizeof(p)) // 8 var s []int = nil fmt.Println(unsafe.Sizeof(s)) // 24 var m map[int]bool = nil fmt.Println(unsafe.Sizeof(m)) // 8 var c chan string = nil fmt.Println(unsafe.Sizeof(c)) // 8 var f func() = nil fmt.Println(unsafe.Sizeof(f)) // 8 var i interface{} = nil fmt.Println(unsafe.Sizeof(i)) // 16 }
不同類型 nil 的指針是一樣的
//不同類型的nil指針是一樣的 package main import ( "fmt" ) func main() { var arr []int var num *int fmt.Printf("%p\n", arr) //0x0 fmt.Printf("%p", num) //0x0 }
通過(guò)運(yùn)行結(jié)果可以看出 arr 和 num 的指針都是 0x0。
不同類型的 nil 是不能比較的
兩個(gè)相同類型的 nil 值也無(wú)法比較
在Go語(yǔ)言中 map、slice 和 function 類型的 nil 值不能比較,比較兩個(gè)無(wú)法比較類型的值是非法的,下面的語(yǔ)句無(wú)法編譯。
但可以將不可比較類型的空值直接與 nil 標(biāo)識(shí)符進(jìn)行比較
//兩個(gè)相同類型的 nil 值也無(wú)法比較 package main import ( "fmt" ) func main() { var s1 []int var s2 []int fmt.Printf(s1 == s2) //invalid operation: s1 == s2 (slice can only be compared to nil) var s3 = []int{1} var s4 = []int{1} var s5 []int copy(s5, s3) fmt.Printf(s3 == s4) //invalid operation: s3 == s4 (slice can only be compared to nil) fmt.Printf(s3 == s5) //invalid operation: s3 == s5 (slice can only be compared to nil) }
對(duì)nil channel,map,slice和array 指針進(jìn)行range操作也是合法的。
- 對(duì)nil map和slice的循環(huán)次數(shù)將是0
- 對(duì)nil數(shù)組的循環(huán)次數(shù)將取決于它的數(shù)組類型定義的長(zhǎng)度
- 對(duì)nil channel的range操作將永遠(yuǎn)阻塞當(dāng)前goroutine
例如,下面的代碼將打印0,1,2,3和4,然后永遠(yuǎn)阻塞。hello, world和bye將永遠(yuǎn)不會(huì)被打印
//對(duì)nil channel,map,slice和array 指針進(jìn)行range操作也是合法的 package main import "fmt" func main() { for range []int(nil) { //循環(huán)次數(shù)將是0 fmt.Println("Hello") } for range map[string]string(nil) { //循環(huán)次數(shù)將是0 fmt.Println("world") } for i := range (*[5]int)(nil) { fmt.Println(i) // 0 1 2 3 4 } for range chan bool(nil) { // block here fmt.Println("Bye") //fatal error: all goroutines are asleep - deadlock! } }
如果類型T的零值是用預(yù)先定義的nil來(lái)表示的話,*new(T)產(chǎn)生一個(gè)nil T類型的值
//如果類型T的零值是用預(yù)先定義的nil來(lái)表示的話,*new(T)產(chǎn)生一個(gè)nil T類型的值 package main import "fmt" func main() { fmt.Println(*new(*int) == nil) // true fmt.Println(*new([]int) == nil) // true fmt.Println(*new(map[int]bool) == nil) // true fmt.Println(*new(chan string) == nil) // true fmt.Println(*new(func()) == nil) // true fmt.Println(*new(interface{}) == nil) // true }
new()返回是一個(gè)指向新分配內(nèi)存的地址,*可以對(duì)地址取值。
以上就是go語(yǔ)言 nil使用避坑指南的詳細(xì)內(nèi)容,更多關(guān)于go語(yǔ)言 nil避坑的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Go語(yǔ)言的代碼組織結(jié)構(gòu)詳細(xì)介紹
這篇文章主要介紹了Go語(yǔ)言的代碼碼組織結(jié)構(gòu)詳細(xì)介紹,即Go語(yǔ)言源碼的文件結(jié)構(gòu),本文講解了包、main和main.main、os包等內(nèi)容,需要的朋友可以參考下2014-10-10golang gin 框架 異步同步 goroutine 并發(fā)操作
這篇文章主要介紹了golang gin 框架 異步同步 goroutine 并發(fā)操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-12-12在Colaboratory上運(yùn)行Go程序的詳細(xì)過(guò)程
這篇文章主要介紹了在Colaboratory上運(yùn)行Go程序,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-08-08Go操作mongodb數(shù)據(jù)庫(kù)方法示例
這篇文章主要為大家介紹了Go操作mongodb數(shù)據(jù)庫(kù)方法示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-09-09基于golang如何實(shí)現(xiàn)error工具包詳解
Go 語(yǔ)言使用 error 類型來(lái)返回函數(shù)執(zhí)行過(guò)程中遇到的錯(cuò)誤,下面這篇文章主要給大家介紹了關(guān)于如何基于golang實(shí)現(xiàn)error工具包的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),需要的朋友可以參考下2018-09-09go語(yǔ)言中布隆過(guò)濾器低空間成本判斷元素是否存在方式
這篇文章主要為大家介紹了go語(yǔ)言中布隆過(guò)濾器低空間成本判斷元素是否存在方式詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-09-09go goroutine 怎樣進(jìn)行錯(cuò)誤處理
在 Go 語(yǔ)言程序開發(fā)中,goroutine 的使用是比較頻繁的,因此在日常編碼的時(shí)候 goroutine 里的錯(cuò)誤處理,怎么做會(huì)比較好呢,本文就來(lái)詳細(xì)介紹一下2021-07-07