不得不知道的golang知識點之nil
來源:互聯網
上載者:User
這是一個建立於 的文章,其中的資訊可能已經有所發展或是發生改變。golang中的`nil`,很多人都誤以為與Java、PHP等程式設計語言中的null一樣。但是實際上Golang的niu複雜得多了,如果不信,那我們繼續往下閱讀。 `nil` 為預聲明的標示符,定義在`builtin/builtin.go`,```// nil is a predeclared identifier representing the zero value for a// pointer, channel, func, interface, map, or slice type.// Type must be a pointer, channel, func, interface, map, or slice typevar nil Type // Type is here for the purposes of documentation only. It is a stand-in// for any Go type, but represents the same type for any given function// invocation.type Type int```## nil的零值按照Go語言規範,任何類型在未初始化時都對應一個零值:布爾類型是false,整型是0,字串是"",而指標、函數、interface、slice、channel和map的零值都是nil。 **PS:這裡沒有說結構體struct的零值為nil,因為struct的零值與其屬性有關** `nil`沒有預設的類型,儘管它是多個類型的零值,必須顯式或隱式指定每個nil用法的明確類型。```package mainfunc main() {// 明確._ = (*struct{})(nil)_ = []int(nil)_ = map[int]bool(nil)_ = chan string(nil)_ = (func())(nil)_ = interface{}(nil)// 隱式.var _ *struct{} = nilvar _ []int = nilvar _ map[int]bool = nilvar _ chan string = nilvar _ func() = nilvar _ interface{} = nil}```如果關注過golang關鍵字的同學就會發現,裡面並沒有`nil`,也就是說`nil`並不是關鍵字,那麼就可以在代碼中定義`nil`,那麼`nil`就會被隱藏。```package mainimport "fmt"func main() {nil := 123fmt.Println(nil) // 123var _ map[string]int = nil //cannot use nil (type int) as type map[string]int in assignment}```## nil類型的地址和值大小`nil`類型的所有值的記憶體布局始終相同,換一句話說就是:不同類型`nil`的記憶體位址是一樣的。```package mainimport ("fmt")func main() {var m map[int]stringvar ptr *intvar sl []intfmt.Printf("%p\n", m) //0x0fmt.Printf("%p\n", ptr ) //0x0fmt.Printf("%p\n", sl ) //0x0}```業務中一般將`nil`值表示為異常。nil值的大小始終與其類型與`nil`值相同的`non-nil`值大小相同。因此, 表示不同零值的nil標識符可能具有不同的大小。```package mainimport ("fmt""unsafe")func main() {var p *struct{} = nilfmt.Println( unsafe.Sizeof( p ) ) // 8var s []int = nilfmt.Println( unsafe.Sizeof( s ) ) // 24var m map[int]bool = nilfmt.Println( unsafe.Sizeof( m ) ) // 8var c chan string = nilfmt.Println( unsafe.Sizeof( c ) ) // 8var f func() = nilfmt.Println( unsafe.Sizeof( f ) ) // 8var i interface{} = nilfmt.Println( unsafe.Sizeof( i ) ) // 16}```大小是編譯器和體繫結構所依賴的。以上列印結果為64位體繫結構和正式 Go 編譯器。對於32位體繫結構, 列印的大小將是一半。 對於正式 Go 編譯器, 同一種類的不同類型的兩個nil值的大小始終相同。例如, 兩個不同的切片類型 ( []int和[]string) 的兩個nil值始終相同。## nil值比較1.不同類型的`nil`是不能比較的。```package mainimport ("fmt")func main() {var m map[int]stringvar ptr *intfmt.Printf(m == ptr) //invalid operation: m == ptr (mismatched types map[int]string and *int)}```在 Go 中, 兩個不同可比較類型的兩個值只能在一個值可以隱式轉換為另一種類型的情況下進行比較。具體來說, 有兩個案例兩個不同的值可以比較:- 兩個值之一的類型是另一個的基礎類型。- 兩個值之一的類型實現了另一個值的類型 (必須是介面類型)。**`nil`值比較並沒有脫離上述規則。**```package mainimport ("fmt")func main() {type IntPtr *intfmt.Println(IntPtr(nil) == (*int)(nil))//truefmt.Println((interface{})(nil) == (*int)(nil))//false}```2.同一類型的兩個`nil`值可能無法比較因為golang中存在map、slice和函數類型是不可比較類型,它們有一個別稱為`不可比擬的類型`,所以比較它們的`nil`亦是非法的。 ```package mainimport ("fmt")func main() {var v1 []int = nilvar v2 []int = nilfmt.Println(v1 == v2)fmt.Println((map[string]int)(nil) == (map[string]int)(nil))fmt.Println((func())(nil) == (func())(nil))}````不可比擬的類型`的值缺是可以與“純nil”進行比較。```package mainimport ("fmt")func main() {fmt.Println((map[string]int)(nil) == nil) //truefmt.Println((func())(nil) == nil) //true}```3.兩`nil`值可能不相等如果兩個比較的nil值之一是一個介面值, 而另一個不是, 假設它們是可比較的, 則比較結果總是 false。原因是在進行比較之前, 介面值將轉換為介面值的類型。轉換後的介面值具有具體的動態類型, 但其他介面值沒有。這就是為什麼比較結果總是錯誤的。```package mainimport ("fmt")func main() {fmt.Println( (interface{})(nil) == (*int)(nil) ) // false}```## 常見問題1.函數返回```func nilReturn() (string,error) {return nil,nil //cannot use nil as type string in return argument}``` 因為`error`是介面類型所以`error`類型沒有報錯。2.map的nil keymap的key為指標、函數、interface、slice、channel和map,則key可以為nil。```package mainimport ("fmt")func main() {mmap := make(map[*string]int,4)a:="a"mmap[&a] = 1mmap[nil] = 99fmt.Println(mmap) //map[0xc042008220:1 <nil>:99]}```## 總結nil之所以比較難以理解因為我們經常混淆了nil值和nil類型,希望各位同學細細品味其中區別。416 次點擊