麻豆小视频在线观看_中文黄色一级片_久久久成人精品_成片免费观看视频大全_午夜精品久久久久久久99热浪潮_成人一区二区三区四区

首頁 > 學院 > 開發設計 > 正文

golang: 詳解interface和nil

2019-11-11 04:49:37
字體:
來源:轉載
供稿:網友

golang的nil在概念上和其它語言的null、None、nil、NULL一樣,都指代零值或空值。nil是預先說明的標識符,也即通常意義上的關鍵字。在golang中,nil只能賦值給指針、channel、func、interface、map或slice類型的變量。如果未遵循這個規則,則會引發panic。對此官方有明確的說明:http://pkg.golang.org/pkg/builtin/#Type

golang中的interface類似于java的interface、php的interface或C++的純虛基類。接口就是一個協議,規定了一組成員。這個沒什么好說的,本文不打算對宏觀上的接口概念和基于接口的范式編程做剖析。golang語言的接口有其獨到之處:只要類型T的公開方法完全滿足接口I的要求,就可以把類型T的對象用在需要接口I的地方。這種做法的學名叫做Structural Typing,有人也把它看作是一種靜態的Duck Typing。所謂類型T的公開方法完全滿足接口I的要求,也即是類型T實現了接口I所規定的一組成員。

在底層,interface作為兩個成員來實現,一個類型和一個值。對此官方也有文檔說明:http://golang.org/doc/go_faq.html#nil_error,如果您不習慣看英文,這里有一篇柴大的翻譯:Go中error類型的nil值和nil 。

接下來通過編寫測試代碼和gdb來看看interface倒底是什么。會用到反射,如果您不太了解golang的反射是什么,這里有刑星翻譯自官方博客的一篇文章:反射的規則,原文在:laws-of-reflection。

[cpp] view plain copy 在CODE上查看代碼片package main     import (      "fmt"      "reflect"  )     func main() {      var val interface{} = int64(58)      fmt.$ cd $GOPATH/src/interface_test  $ go build -gcflags "-N -l"  $ gdb interface_test  

接下來說說interface類型的值和nil的比較問題。這是個比較經典的問題,也算是golang的一個坑。

接下來看看代碼

[cpp] view plain copy 在CODE上查看代碼片package main  import (      "fmt"  )  func main() {      var val interface{} = nil      if val == nil {          fmt.Println("val is nil")      } else {          fmt.Println("val is not nil")      }  }  變量val是interface類型,它的底層結構必然是(type,%20data)。由于nil是untyped(無類型),而又將nil賦值給了變量val,所以val實際上存儲的是(nil,%20nil)。因此很容易就知道val和nil的相等比較是為true的。

[cpp] view%20plain copy $ cd $GOPATH/src/interface_test  $ go build  $ ./interface_test  val is nil  

對于將任何其它有意義的值類型賦值給val,都導致val持有一個有效的類型和數據。也就是說變量val的底層結構肯定不為(nil,%20nil),因此它和nil的相等比較總是為false。

上面的討論都是在圍繞值類型來進行的。在繼續討論之前,讓我們來看一種特例:(*interface{})(nil)。將nil轉成interface類型的指針,其實得到的結果僅僅是空接口類型指針并且它指向無效的地址。注意是空接口類型指針而不是空指針,這兩者的區別蠻大的,學過C的童鞋都知道空指針是什么概念。

關于(*interface{})(nil)還有一些要注意的地方。這里僅僅是拿(*interface{})(nil)來舉例,對于(*int)(nil)、(*byte)(nil)等等來說是一樣的。上面的代碼定義了接口指針類型變量val,它指向無效的地址(0x0),因此val持有無效的數據。但它是有類型的(*interface{})。所以val的底層結構應該是:(*interface{},%20nil)。有時候您會看到(*interface{})(nil)的應用,比如var%20ptrIface%20=%20(*interface{})(nil),如果您接下來將ptrIface指向其它類型的指針,將通不過編譯。或者您這樣賦值:*ptrIface%20=%20123,那樣的話編譯是通過了,但在運行時還是會panic的,這是因為ptrIface指向的是無效的內存地址。其實聲明類似ptrIface這樣的變量,是因為使用者只是關心指針的類型,而忽略它存儲的值是什么。還是以例子來說明:

[cpp] view%20plain copy package main  import (      "fmt"  )  func main() {      var val interface{} = (*interface{})(nil)      // val = (*int)(nil)      if val == nil {          fmt.Println("val is nil")      } else {          fmt.Println("val is not nil")  很顯然,無論該指針的值是什么:(*interface{},%20nil),這樣的接口值總是非nil的,即使在該指針的內部為nil。

[cpp] view%20plain copy $ cd $GOPATH/src/interface_test  $ go build  $ ./interface_test  val is not nil  interface類型的變量和nil的相等比較出現最多的地方應該是error接口類型的值與nil的比較。有時候您想自定義一個返回錯誤的函數來做這個事,可能會寫出以下代碼:

[cpp] view%20plain copy package main     import (      "fmt"  )     type data struct{}     func (this *data) Error() string { return "" }     func test() error {      var p *data = nil      return p  }     func main() {      var e error = test()      if e == nil {          fmt.Println("e is nil")      } else {          fmt.Println("e is not nil")      }  }  但是很可惜,以上代碼是有問題的。

[cpp] view%20plain copy $ cd $GOPATH/src/interface_test  $ go build  $ ./interface_test  e is not nil  

我們可以來分析一下。error是一個接口類型,test方法中返回的指針p雖然數據是nil,但是由于它被返回成包裝的error類型,也即它是有類型的。所以它的底層結構應該是(*data,%20nil),很明顯它是非nil的。可以打印觀察下底層結構數據:

[cpp] view%20plain copy package main     import (      "fmt"      "unsafe"  )     type data struct{}     func (this *data) Error() string { return "" }     func test() error {      var p *data = nil      return p  }     func main() {      var e error = test()         d := (*struct {          itab uintptr          data uintptr      })(unsafe.Pointer(&e))         fmt.Println(d)  }    $ cd $GOPATH/src/interface_test  $ go build  $ ./interface_test  &{3078907912 0}  正確的做法應該是:

[cpp] view%20plain copy 派生到我的代碼片package main     import (      "fmt"  )     type data struct{}     func (this *data) Error() string { return "" }     func bad() bool {      return true  }     func test() error {      var p *data = nil      if bad() {          return p      }      return nil  }     func main() {      var e error = test()      if e == nil {          fmt.Println("e is nil")      } else {          fmt.Println("e is not nil")      }  }  


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 女人裸体让男人桶全过程 | 久草手机在线 | 久久久久久久久免费 | 国产成人精品二区 | 免费黄色欧美视频 | 国产精品成人免费一区久久羞羞 | 久久久一区二区三区视频 | 亚洲涩涩图 | 深夜福利久久久 | 日本一区二区视频在线观看 | 黄色片网站免费观看 | 欧美一级黄色录相 | 日本免费不卡一区二区 | 黄色片一区二区 | 男男羞羞视频网站国产 | 毛片在线免费视频 | 成人444kkkk在线观看 | 在线中文字幕观看 | 狠狠干91 | 欧美成人三级大全 | 欧美精品成人一区二区三区四区 | 久久91久久 | 久久另类视频 | 成人黄色小视频在线观看 | 欧洲黄视频 | 久久精品中文字幕一区二区三区 | 狠狠干天天操 | 国产亚洲精彩视频 | 亚州精品国产 | 青青国产在线视频 | 91精品国产乱码久久桃 | 成人在线视频一区 | 国产91久久久 | 青青草免费观看完整版高清 | av电影在线观看网站 | asian超清日本肉体pics | 九九热这里只有精品8 | 免费视频a | 欧美性受xxxxxx黑人xyx性爽 | 国产精品99久久久久久久 | 国产精品99久久久久久宅女 |