首頁 > 軟體

go語言 nil使用避坑指南

2022-09-02 18:02:35

引言

今天筆試題遇到 var x string = nil ,問這個定義是否正確?

這裡給出答案:

cannot use nil as string value in variable declaration。

也就是說,string型別和nil八竿子打不著,要想判斷字串是否為空,可以使用str == ""或者len(str) == 0

接下來,順便總結一下nil的使用

nil

nil 是go語言中預先定義的識別符號,不是關鍵字或保留字。 我們可以直接使用nil,而不用宣告它。 而且我們可以定義一個名稱為 nil 的變數,比如下面這樣:

var nil = errors.New("nil")
fmt.Printf("%#vn", nil)//&errors.errorString{s:"nil"}

雖然上面的宣告語句可以通過編譯,但是並不提倡這麼做。

預設值nil (重點記住)

在go語言中:

  • 布林型別的零值(初始值)為 false
  • 數值型別的零值為 0
  • 字串型別的零值為空字串""

除此之外其它型別的預設值為nilnil可以代表下面這些型別的零值:

  • 指標型別(包括unsafe中的)
  • map型別
  • slice型別
  • function型別
  • channel型別
  • interface型別

nil沒有預設型別

預先定義的nil是唯一的一個go語言中沒有預設型別的非型別值。對於編譯器來說,必須從上下文中獲取充足的資訊才能推斷出nil的型別。

當你把nil賦值給一個channel型別變數,此時為channel型別。

當你把nil賦值給map型別變數,此時為map型別。

不同型別的nil值佔用的記憶體大小可能是不一樣的

一個型別的所有的值的記憶體佈局都是一樣的。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("%pn", arr)    //0x0
   fmt.Printf("%p", num)  //0x0
}

通過執行結果可以看出 arr 和 num 的指標都是 0x0。

不同型別的 nil 是不能比較的

兩個相同型別的 nil 值也無法比較

在Go語言中 map、slice 和 function 型別的 nil 值不能比較,比較兩個無法比較型別的值是非法的,下面的語句無法編譯。

但可以將不可比較型別的空值直接與 nil 識別符號進行比較

//兩個相同型別的 nil 值也無法比較
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)
}

對nil channel,map,slice和array 指標進行range操作也是合法的。

  • 對nil map和slice的迴圈次數將是0
  • 對nil陣列的迴圈次數將取決於它的陣列型別定義的長度
  • 對nil channel的range操作將永遠阻塞當前goroutine

例如,下面的程式碼將列印0,1,2,3和4,然後永遠阻塞。hello, world和bye將永遠不會被列印

//對nil channel,map,slice和array 指標進行range操作也是合法的
package main
import "fmt"
func main() {
   for range []int(nil) { //迴圈次數將是0
      fmt.Println("Hello")
   }
   for range map[string]string(nil) { //迴圈次數將是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的零值是用預先定義的nil來表示的話,*new(T)產生一個nil T型別的值

//如果型別T的零值是用預先定義的nil來表示的話,*new(T)產生一個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()返回是一個指向新分配記憶體的地址,*可以對地址取值。

以上就是go語言 nil使用避坑指南的詳細內容,更多關於go語言 nil避坑的資料請關注it145.com其它相關文章!


IT145.com E-mail:sddin#qq.com