在 Go 語言中,布爾型別的零值(初始值)為 false,數值型別的零值為 0,字串型別的零值為空字串"",而指標、切片、映射、通道、函式和介面的零值則是 nil,
nil 是Go語言中一個預定義好的識別符號,有過其他編程語言開發經驗的開發者也許會把 nil 看作其他語言中的 null(NULL),其實這并不是完全正確的,因為Go語言中的 nil 和其他語言中的 null 有很多不同點,
下面通過幾個方面來介紹一下Go語言中 nil,
nil 識別符號是不能比較的
package mainimport ( "fmt")func main() { fmt.Println(nil==nil)}運行結果如下所示:
PS D:\code> go run .\main.go# command-line-arguments.\main.go:8:21: invalid operation: nil == nil (operator == not defined on nil)//這點和 python 等動態語言是不同的,在 python 中,兩個 None 值永遠相等,>>> None == NoneTrue
從上面的運行結果不難看出,==對于 nil 來說是一種未定義的操作,
nil 不是關鍵字或保留字
nil 并不是Go語言的關鍵字或者保留字,也就是說我們可以定義一個名稱為 nil 的變數,比如下面這樣:
var nil = errors.New("my god")雖然上面的宣告陳述句可以通過編譯,但是并不提倡這么做,
nil 沒有默認型別
package mainimport ( "fmt")func main() { fmt.Printf("%T", nil) print(nil)}運行結果如下所示:
PS D:\code> go run .\main.go# command-line-arguments.\main.go:9:10: use of untyped nil
不同型別 nil 的指標是一樣的
package mainimport ( "fmt")func main() { var arr []int var num *int fmt.Printf("%p\n", arr) fmt.Printf("%p", num)}運行結果如下所示:
PS D:\code> go run .\main.go0x00x0
通過運行結果可以看出 arr 和 num 的指標都是 0x0,
不同型別的 nil 是不能比較的
package mainimport ( "fmt")func main() { var m map[int]string var ptr *int fmt.Printf(m == ptr)}運行結果如下所示:
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 mainimport ( "fmt")func main() { var s1 []int var s2 []int fmt.Printf(s1 == s2)}運行結果如下所示:
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)
通過上面的錯誤提示可以看出,能夠將上述不可比較型別的空值直接與 nil 識別符號進行比較,如下所示:
package mainimport ( "fmt")func main() { var s1 []int fmt.Println(s1 == nil)}nil 是 map、slice、pointer、channel、func、interface 的零值
package mainimport ( "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)}運行結果如下所示:
PS D:\code> go run .\main.gomap[int]string(nil)(*int)(nil)(chan int)(nil)[]int(nil)(func())(nil)<nil>
零值是Go語言中變數在宣告之后但是未初始化被賦予的該型別的一個默認值,
不同型別的 nil 值占用的記憶體大小可能是不一樣的
一個型別的所有的值的記憶體布局都是一樣的,nil 也不例外,nil 的大小與同型別中的非 nil 型別的大小是一樣的,但是不同型別的 nil 值的大小可能不同,package mainimport ( "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}運行結果如下所示:
PS D:\code> go run .\main.go82488816具體的大小取決于編譯器和架構,上面列印的結果是在 64 位架構和標準編譯器下完成的,對應 32 位的架構的,列印的大小將減半,
轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/56901.html
標籤:Go
