Go語言nil標(biāo)識符(空值/零值)
在Go語言中,布爾類型的零值(初始值)為 false,數(shù)值類型的零值為 0,字符串類型的零值為空字符串"",而指針、切片、映射、通道、函數(shù)和接口的零值則是 nil。
nil 是Go語言中一個預(yù)定義好的標(biāo)識符,有過其他編程語言開發(fā)經(jīng)驗的開發(fā)者也許會把 nil 看作其他語言中的 null(NULL),其實這并不是完全正確的,因為Go語言中的 nil 和其他語言中的 null 有很多不同點。
下面通過幾個方面來介紹一下Go語言中 nil。
nil 標(biāo)識符是不能比較的
package main import ( "fmt" ) func main() { fmt.Println(nil==nil) }
運行結(jié)果如下所示:
PS D:\code> go run .\main.go
# command-line-arguments
.\main.go:8:21: invalid operation: nil == nil (operator == not defined on nil)
這點和 python 等動態(tài)語言是不同的,在 python 中,兩個 None 值永遠(yuǎn)相等。
>>> None == None True
從上面的運行結(jié)果不難看出,==對于 nil 來說是一種未定義的操作。
nil 不是關(guān)鍵字或保留字
nil 并不是Go語言的關(guān)鍵字或者保留字,也就是說我們可以定義一個名稱為 nil 的變量,比如下面這樣:
var nil = errors.New("my god")
雖然上面的聲明語句可以通過編譯,但是并不提倡這么做。
nil 沒有默認(rèn)類型
package main import ( "fmt" ) func main() { fmt.Printf("%T", nil) print(nil) }
運行結(jié)果如下所示:
PS D:\code> go run .\main.go
# command-line-arguments
.\main.go:9:10: use of untyped nil
不同類型 nil 的指針是一樣的
package main import ( "fmt" ) func main() { var arr []int var num *int fmt.Printf("%p\n", arr) fmt.Printf("%p", num) }
運行結(jié)果如下所示:
PS D:\code> go run .\main.go
0x0
0x0
通過運行結(jié)果可以看出 arr 和 num 的指針都是 0x0。
不同類型的 nil 是不能比較的
package main import ( "fmt" ) func main() { var m map[int]string var ptr *int fmt.Printf(m == ptr) }
運行結(jié)果如下所示:
PS D:\code> go run .\main.go
# command-line-arguments
.\main.go:10:20: invalid operation: arr == ptr (mismatched types []int and *int)
兩個相同類型的 nil 值也可能無法比較
在Go語言中 map、slice 和 function 類型的 nil 值不能比較,比較兩個無法比較類型的值是非法的,下面的語句無法編譯。
package main import ( "fmt" ) func main() { var s1 []int var s2 []int fmt.Printf(s1 == s2) }
運行結(jié)果如下所示:
PS D:\code> go run .\main.go
# command-line-arguments
.\main.go:10:19: invalid operation: s1 == s2 (slice can only be compared to nil)
通過上面的錯誤提示可以看出,能夠?qū)⑸鲜霾豢杀容^類型的空值直接與 nil 標(biāo)識符進(jìn)行比較,如下所示:
package main import ( "fmt" ) func main() { var s1 []int fmt.Println(s1 == nil) }
運行結(jié)果如下所示:
PS D:\code> go run .\main.go
true
nil 是 map、slice、pointer、channel、func、interface 的零值
package main import ( ? ? "fmt" ) func main() { ? ? var m map[int]string ? ? var ptr *int ? ? var c chan int ? ? var sl []int ? ? var f func() ? ? var i interface{} ? ? fmt.Printf("%#v\n", m) ? ? fmt.Printf("%#v\n", ptr) ? ? fmt.Printf("%#v\n", c) ? ? fmt.Printf("%#v\n", sl) ? ? fmt.Printf("%#v\n", f) ? ? fmt.Printf("%#v\n", i) }
運行結(jié)果如下所示:
PS D:\code> go run .\main.go
map[int]string(nil)
(*int)(nil)
(chan int)(nil)
[]int(nil)
(func())(nil)
<nil>
零值是Go語言中變量在聲明之后但是未初始化被賦予的該類型的一個默認(rèn)值。
不同類型的 nil 值占用的內(nèi)存大小可能是不一樣的
一個類型的所有的值的內(nèi)存布局都是一樣的,nil 也不例外,nil 的大小與同類型中的非 nil 類型的大小是一樣的。但是不同類型的 nil 值的大小可能不同。
package main import ( "fmt" "unsafe" ) func main() { var p *struct{} fmt.Println( unsafe.Sizeof( p ) ) // 8 var s []int fmt.Println( unsafe.Sizeof( s ) ) // 24 var m map[int]bool fmt.Println( unsafe.Sizeof( m ) ) // 8 var c chan string fmt.Println( unsafe.Sizeof( c ) ) // 8 var f func() fmt.Println( unsafe.Sizeof( f ) ) // 8 var i interface{} fmt.Println( unsafe.Sizeof( i ) ) // 16 }
運行結(jié)果如下所示:
PS D:\code> go run .\main.go
8
24
8
8
8
16
具體的大小取決于編譯器和架構(gòu),上面打印的結(jié)果是在 64 位架構(gòu)和標(biāo)準(zhǔn)編譯器下完成的,對應(yīng) 32 位的架構(gòu)的,打印的大小將減半。
到此這篇關(guān)于Go語言nil標(biāo)識符(空值/零值)的文章就介紹到這了,更多相關(guān)Go語言nil內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
go語言實現(xiàn)將重要數(shù)據(jù)寫入圖片中
本文給大家分享的是go語言實現(xiàn)將數(shù)據(jù)的二進(jìn)制形式寫入圖像紅色通道數(shù)據(jù)二進(jìn)制的低位,從而實現(xiàn)將重要數(shù)據(jù)隱藏,有需要的小伙伴參考下吧。2015-03-03go slice 擴(kuò)容實現(xiàn)原理源碼解析
這篇文章主要為大家介紹了go slice 擴(kuò)容實現(xiàn)原理源碼解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-01-01