婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av

主頁 > 知識庫 > 基于go interface{}==nil 的幾種坑及原理分析

基于go interface{}==nil 的幾種坑及原理分析

熱門標(biāo)簽:涿州代理外呼系統(tǒng) excel地圖標(biāo)注分布數(shù)據(jù) 壽光微信地圖標(biāo)注 阿克蘇地圖標(biāo)注 評價(jià)高的400電話辦理 外呼系統(tǒng)顯本地手機(jī)號 外呼系統(tǒng)用什么卡 電話機(jī)器人軟件免費(fèi) 百度地圖標(biāo)注后傳給手機(jī)

本文是Go比較有名的一個(gè)坑,在以前面試的時(shí)候也被問過,為什么想起來寫這個(gè)?

因?yàn)槲覀兙€上就真實(shí)出現(xiàn)過這個(gè)坑,寫給不了解的人在使用 if err != nil 的時(shí)候提高警惕。

Go語言的interface{}在使用過程中有一個(gè)特別坑的特性,當(dāng)你比較一個(gè)interface{}類型的值是否是nil的時(shí)候,這是需要特別注意避免的問題。

先來看看一個(gè)demo:

package main
import "fmt"
type ErrorImpl struct{}
func (e *ErrorImpl) Error() string {
   return ""
}
var ei *ErrorImpl
var e error
func ErrorImplFun() error {
   return ei
}
func main() {
   f := ErrorImplFun()
   fmt.Println(f == nil)
}

輸出:

false

為什么不是true?

想要理解這個(gè)問題,首先需要理解interface{}變量的本質(zhì)。在Go語言中,一個(gè)interface{}類型的變量包含了2個(gè)指針,一個(gè)指針指向值的在編譯時(shí)確定的類型,另外一個(gè)指針指向?qū)嶋H的值。

// InterfaceStructure 定義了一個(gè)interface{}的內(nèi)部結(jié)構(gòu)
type InterfaceStructure struct {
  pt uintptr // 到值類型的指針
  pv uintptr // 到值內(nèi)容的指針
}
// asInterfaceStructure 將一個(gè)interface{}轉(zhuǎn)換為InterfaceStructure
func asInterfaceStructure(i interface{}) InterfaceStructure {
  return *(*InterfaceStructure)(unsafe.Pointer(i))
}
func main() {
  var i1, i2 interface{}
  var v1 int = 23
  var v2 int = 23
  i1 = v1
  i2 = v2
  fmt.Printf("sizeof interface{} = %d\n", unsafe.Sizeof(i1))
  fmt.Printf("i1 %v %+v\n", i1, asInterfaceStructure(i1))
  fmt.Printf("i2 %v %+v\n", i2, asInterfaceStructure(i2))
  var nilInterface interface{}
  var str *string
  fmt.Printf("nil interface = %+v\n", asInterfaceStructure(nilInterface))
  fmt.Printf("nil string = %+v\n", asInterfaceStructure(str))
  fmt.Printf("nil = %+v\n", asInterfaceStructure(nil))
}

輸出:

sizeof interface{} = 16

i1 23 {pt:4812032 pv:825741246928}

i2 23 {pt:4812032 pv:825741246936}

nil interface = {pt:0 pv:0}

nil string = {pt:4802400 pv:0}

nil = {pt:0 pv:0}

當(dāng)我們將一個(gè)具體類型的值賦值給一個(gè)interface{}類型的變量的時(shí)候,就同時(shí)把類型和值都賦值給了interface{}里的兩個(gè)指針。如果這個(gè)具體類型的值是nil的話,interface{}變量依然會存儲對應(yīng)的類型指針和值指針。

如何解決?

方法一

返回的結(jié)果進(jìn)行非nil檢查,然后再賦值給interface{}變量

type ErrorImpl struct{}
func (e *ErrorImpl) Error() string {
   return ""
}
var ei *ErrorImpl
var e error
func ErrorImplFun() error {
   if ei == nil {
      return nil
   }
   return ei
}
func main() {
   f := ErrorImplFun()
   fmt.Println(f == nil)
}

輸出:

true

方法二

返回具體實(shí)現(xiàn)的類型而不是interface{}

package main
import "fmt"
type ErrorImpl struct{}
func (e *ErrorImpl) Error() string {
   return ""
}
var ei *ErrorImpl
var e error
func ErrorImplFun() *ErrorImpl {
   return ei
}
func main() {
   f := ErrorImplFun()
   fmt.Println(f == nil)
}

輸出:

true

解決由于第三方包帶來的坑

由于有的error是第三方包返回的,又自己不想改第三方包,只好接收處理的時(shí)候想辦法。

方法一

利用interface{}原理

 is:=*(*InterfaceStructure)(unsafe.Pointer(i))
 if is.pt==0  is.pv==0 {
     //is nil do something
 }

將底層指向值和指向值的類型的指針打印出來如果都是0,表示是nil

方法二

利用斷言,斷言出來具體類型再判斷非空

type ErrorImpl struct{}
func (e ErrorImpl) Error() string {
   return "demo"
}
var ei *ErrorImpl
var e error
func ErrorImplFun() error {
   //ei = ErrorImpl{}
   return ei
}
func main() {
   f := ErrorImplFun()
   //當(dāng)然error實(shí)現(xiàn)類型較多的話使用  
 //switch case方式斷言更清晰
   res, ok := f.(*ErrorImpl)
   fmt.Printf("ok:%v,f:%v,res:%v", 
   ok, f == nil, res == nil)
}

輸出:

ok:true,f:false,res:true

方法三

利用反射

type ErrorImpl struct{}
func (e ErrorImpl) Error() string {
   return "demo"
}
var ei *ErrorImpl
var e error
func ErrorImplFun() error {
   //ei = ErrorImpl{}
   return ei
}
func main() {
   f := ErrorImplFun()
   rv := reflect.ValueOf(f)
   fmt.Printf("%v", rv.IsNil())
}

輸出:

true

注意⚠:

斷言和反射性能不是特別好,如果不得已再使用,控制使用有助于提升程序性能。

由于函數(shù)接收類型導(dǎo)致的panic:

type ErrorImpl struct{}
func (e ErrorImpl) Error() string {
   return "demo"
}
var ei *ErrorImpl
var e error
func ErrorImplFun() error {
   return ei
}
func main() {
   f := ErrorImplFun()
   fmt.Printf(f.Error())
}

輸出:

panic: value method main.ErrorImpl.Error called using nil *ErrorImpl pointer

解決:

func (e *ErrorImpl) Error() string {
   return "demo"
}

輸出:

demo

可以發(fā)現(xiàn)將接收類型變成指針類型就可以了。

以上就是 nil 相關(guān)的坑,希望大家可以牢記,如果 ”幸運(yùn)“ 的遇到了,可以想到這些可能性。

補(bǔ)充:go 語言 interface{} 的易錯(cuò)點(diǎn)

如果說 goroutine 和 channel 是 go 語言并發(fā)的兩大基石,那 interface 就是 go 語言類型抽象的關(guān)鍵。

在實(shí)際項(xiàng)目中,幾乎所有的數(shù)據(jù)結(jié)構(gòu)最底層都是接口類型。

說起 C++ 語言,我們立即能想到是三個(gè)名詞:封裝、繼承、多態(tài)。go 語言雖然沒有嚴(yán)格意義上的對象,但通過 interface,可以說是實(shí)現(xiàn)了多態(tài)性。(由以組合結(jié)構(gòu)體實(shí)現(xiàn)了封裝、繼承的特性)

package main
type animal interface {
    Move()
}
type bird struct{}
func (self *bird) Move() {
    println("bird move")
}
type beast struct{}
func (self *beast) Move() {
    println("beast move")
}
func animalMove(v animal) {
    v.Move()
}
func main() {
    var a *bird
    var b *beast
    animalMove(a) // bird move
    animalMove(b) // beast move
}

go 語言中支持將 method、struct、struct 中成員定義為 interface 類型,使用 struct 舉一個(gè)簡單的栗子

使用 go 語言的 interface 特性,就能實(shí)現(xiàn)多態(tài)性,進(jìn)行泛型編程。

二,interface 原理

如果沒有充分了解 interface 的本質(zhì),就直接使用,那最終肯定會踩到很深的坑,要用就先要了解,先來看看 interface 源碼

 type eface struct {
     _type *_type
     data  unsafe.Pointer
 }  
 type _type struct {
     size       uintptr // type size
     ptrdata    uintptr // size of memory prefix holding all pointers
     hash       uint32  // hash of type; avoids computation in hash tables
     tflag      tflag   // extra type information flags
     align      uint8   // alignment of variable with this type
     fieldalign uint8   // alignment of struct field with this type
     kind       uint8   // enumeration for C
     alg        *typeAlg  // algorithm table
     gcdata    *byte    // garbage collection data
     str       nameOff  // string form
     ptrToThis typeOff  // type for pointer to this type, may be zero
 }

可以看到 interface 變量之所以可以接收任何類型變量,是因?yàn)槠浔举|(zhì)是一個(gè)對象,并記錄其類型和數(shù)據(jù)塊的指針。(其實(shí) interface 的源碼還包含函數(shù)結(jié)構(gòu)和內(nèi)存分布,由于不是本文重點(diǎn),有興趣的同學(xué)可以自行了解)

三,interface 判空的坑

對于一個(gè)空對象,我們往往通過 if v == nil 的條件語句判斷其是否為空,但在代碼中充斥著 interface 類型的情況下,很多時(shí)候判空都并不是我們想要的結(jié)果(其實(shí)了解或聰明的同學(xué)從上述 interface 的本質(zhì)是對象已經(jīng)知道我想要說的是什么)

package main 
 type animal interface {
     Move()
 } 
 type bird struct{} 
 func (self *bird) Move() {
     println("bird move")
 } 
 type beast struct{} 
 func (self *beast) Move() {
     println("beast move")
 } 
 func animalMove(v animal) {
     if v == nil {
         println("nil animal")
     }
     v.Move()
 } 
 func main() {
     var a *bird   // nil
     var b *beast  // nil
     animalMove(a) // bird move
     animalMove(b) // beast move
 }

還是剛才的栗子,其實(shí)在 go 語言中 var a *bird 這種寫法,a 只是聲明了其類型,但并沒有申請一塊空間,所以這時(shí)候 a 本質(zhì)還是指向空指針,但我們在 aminalMove 函數(shù)進(jìn)行判空是失敗的,并且下面的 v.Move() 的調(diào)用也是成功的,本質(zhì)的原因就是因?yàn)?interface 是一個(gè)對象,在進(jìn)行函數(shù)調(diào)用的時(shí)候,就會將 bird 類型的空指針進(jìn)行隱式轉(zhuǎn)換,轉(zhuǎn)換成實(shí)例的 interface animal 對象,所以這時(shí)候 v 其實(shí)并不是空,而是其 data 變量指向了空。

這時(shí)候看著執(zhí)行都正常,那什么情況下坑才會絆倒我們呢?只需要加一段代碼

package main 
 type animal interface {
     Move()
 } 
 type bird struct {
    name string
 } 
 func (self *bird) Move() {
     println("bird move %s", self.name) // panic
 } 
 type beast struct {
     name string
 } 
 func (self *beast) Move() {
     println("beast move %s", self.name) // panic
 } 
 func animalMove(v animal) {
     if v == nil {
         println("nil animal")
     }
     v.Move()
 } 
 func main() {
     var a *bird   // nil
     var b *beast  // nil
     animalMove(a) // panic
     animalMove(b) // panic
 }

在代碼中,我們給派生類添加 name 變量,并在函數(shù)的實(shí)現(xiàn)中進(jìn)行調(diào)用,就會發(fā)生 panic,這時(shí)候的 self 其實(shí)是 nil 指針。所以這里坑就出來了。

有些人覺得這類錯(cuò)誤謹(jǐn)慎一些還是可以避免的,那是因?yàn)槲覀兪钦蛩季S去代入接口,但如果反向編程就容易造成很難發(fā)現(xiàn)的 bug

package main 
 type animal interface {
     Move()
 } 
 type bird struct {
     name string
 } 
 func (self *bird) Move() {
     println("bird move %s", self.name)
 } 
 type beast struct {
     name string
 } 
 func (self *beast) Move() {
     println("beast move %s", self.name)
 } 
 func animalMove(v animal) {
     if v == nil {
         println("nil animal")
     }
     v.Move()
 } 
 func getBirdAnimal(name string) *bird {
     if name != "" {
         return bird{name: name}
     }
     return nil
 } 
 func main() {
     var a animal
     var b animal
     a = getBirdAnimal("big bird")
     b = getBirdAnimal("") // return interface{data:nil}
     animalMove(a) // bird move big bird
     animalMove(b) // panic
 }

這里我們看到通過函數(shù)返回實(shí)例類型指針,當(dāng)返回 nil 時(shí),因?yàn)榻邮盏淖兞繛榻涌陬愋停赃M(jìn)行了隱性轉(zhuǎn)換再次導(dǎo)致了 panic(這類反向轉(zhuǎn)換很難發(fā)現(xiàn))。

那我們?nèi)绾翁幚砩鲜鲞@類問題呢。我這邊整理了三個(gè)點(diǎn)

1,充分了解 interface 原理,使用過程中需要謹(jǐn)慎小心

2,謹(jǐn)慎使用泛型編程,接收變量使用接口類型,也需要保證接口返回為接口類型,而不應(yīng)該是實(shí)例類型

3,判空是使用反射 typeOf 和 valueOf 轉(zhuǎn)換成實(shí)例對象后再進(jìn)行判空

您可能感興趣的文章:
  • Golang中interface{}轉(zhuǎn)為數(shù)組的操作
  • 淺談Golang 嵌套 interface 的賦值問題
  • Golang 實(shí)現(xiàn)interface類型轉(zhuǎn)string類型
  • 解決golang 反射interface{}做零值判斷的一個(gè)重大坑
  • golang interface判斷為空nil的實(shí)現(xiàn)代碼
  • 詳解Golang語言中的interface
  • 使用go的interface案例實(shí)現(xiàn)多態(tài)范式操作
  • go 類型轉(zhuǎn)換方式(interface 類型的轉(zhuǎn)換)

標(biāo)簽:重慶 銅川 梅河口 吐魯番 蘭州 汕頭 欽州 雞西

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《基于go interface{}==nil 的幾種坑及原理分析》,本文關(guān)鍵詞  基于,interface,nil,的,幾種,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《基于go interface{}==nil 的幾種坑及原理分析》相關(guān)的同類信息!
  • 本頁收集關(guān)于基于go interface{}==nil 的幾種坑及原理分析的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章
    婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av
    国产精品亲子乱子伦xxxx裸| 午夜欧美一区二区三区在线播放| 亚洲成人黄色影院| 艳妇臀荡乳欲伦亚洲一区| 日韩成人av影视| 在线视频亚洲一区| 日本一二三不卡| 蜜桃久久精品一区二区| 色噜噜狠狠成人中文综合| 亚洲国产精品成人久久综合一区| 日韩国产在线观看一区| 国产精品一区二区在线看| 欧美裸体bbwbbwbbw| 一区二区三区中文字幕精品精品| 国产成人夜色高潮福利影视| 欧美日韩电影在线播放| 三级影片在线观看欧美日韩一区二区 | 美女网站色91| 91 com成人网| 日韩制服丝袜av| 久久综合国产精品| 成人av电影在线网| 天堂成人国产精品一区| 精品粉嫩超白一线天av| 国产激情一区二区三区四区| 亚洲视频资源在线| 日韩女优电影在线观看| 成人免费视频视频| 日本中文字幕一区二区视频 | 蜜臀av性久久久久蜜臀av麻豆| 欧美中文字幕一区二区三区| 久久国产精品色| 国产精品初高中害羞小美女文| 色婷婷久久久久swag精品| 精彩视频一区二区| 有坂深雪av一区二区精品| 欧美zozozo| 欧美一区二区三区日韩| 色哟哟一区二区三区| 国产一区二区成人久久免费影院| 亚洲电影你懂得| 亚洲精品少妇30p| 国产精品视频一二三区| 久久亚洲欧美国产精品乐播 | 久久久久久久精| 91精品国产麻豆国产自产在线| 99久久精品免费看国产| 精品一区二区三区不卡| 免费不卡在线观看| 蜜臀av在线播放一区二区三区| 一区二区三区精品| 中文字幕一区av| 国产欧美一区视频| 精品国产一区二区三区av性色 | 精品国产凹凸成av人导航| 亚洲精品老司机| 国产精品久久久久久久久免费樱桃| 3d动漫精品啪啪1区2区免费| 日本电影欧美片| 欧美美女一区二区在线观看| 91麻豆精品国产无毒不卡在线观看 | 婷婷开心久久网| 精品一区二区三区香蕉蜜桃| 国产一区欧美二区| 国产精品中文字幕一区二区三区| 国产一区二区三区| 91丝袜高跟美女视频| 精品视频一区三区九区| 日韩欧美国产综合| 国产女人18水真多18精品一级做 | 中文字幕一区av| 午夜久久久久久电影| 久久99热国产| 欧美无砖砖区免费| 国产女人18毛片水真多成人如厕| 亚洲欧美日韩国产综合| 日本不卡在线视频| 天堂久久久久va久久久久| 国产成人综合视频| 欧美日韩精品欧美日韩精品一综合| 久久先锋资源网| 亚洲午夜av在线| 国产91丝袜在线播放0| 欧美精品粉嫩高潮一区二区| 国产精品第五页| 一本一道波多野结衣一区二区| 韩国精品在线观看| 欧美人与禽zozo性伦| 中文字幕在线观看不卡| 蜜臀av性久久久久蜜臀aⅴ流畅 | 亚洲免费观看高清完整版在线观看 | 精品一区二区三区蜜桃| 91片黄在线观看| 亚洲另类在线一区| 成人av网站在线| 国产精品久久久久久久久果冻传媒| 麻豆一区二区三区| 欧美日韩精品欧美日韩精品| 亚洲精品欧美激情| 成人污污视频在线观看| 最近日韩中文字幕| 欧美亚洲国产一区二区三区va| 亚洲永久精品大片| 欧美日韩国产高清一区二区| 美腿丝袜亚洲一区| 日韩一区二区在线观看视频播放| 日韩国产在线观看| 欧美激情综合五月色丁香| 成人黄页在线观看| 亚洲高清中文字幕| 久久免费国产精品| 亚洲第一福利视频在线| 欧美韩日一区二区三区四区| 国产精品久久久久一区二区三区共 | 26uuu精品一区二区| 99久久国产综合色|国产精品| 夜夜精品视频一区二区| 欧美大尺度电影在线| 91香蕉视频黄| 国产一区二区不卡在线| 亚洲成人免费看| 国产精品久久精品日日| 欧美一区二区大片| 欧美在线视频全部完| 丁香婷婷综合色啪| 精品无码三级在线观看视频| 亚洲国产日韩一级| 国产欧美日韩不卡免费| 欧美另类videos死尸| 欧美自拍偷拍一区| 99久久99久久精品国产片果冻| 久久国产精品色| 日本中文在线一区| 久久欧美中文字幕| 日韩欧美一区电影| 日韩视频免费观看高清在线视频| 在线观看国产一区二区| 99re热视频精品| 91一区二区在线观看| 成人免费毛片嘿嘿连载视频| 精品影院一区二区久久久| 美腿丝袜亚洲一区| 久久精品久久久精品美女| 久久国产精品一区二区| 国产在线精品不卡| 日韩精品国产精品| 首页综合国产亚洲丝袜| 日本成人在线网站| 久久99精品久久久久久国产越南| 亚洲成人777| 精品一区二区三区久久| 99久久久国产精品| 在线精品视频小说1| 91久久人澡人人添人人爽欧美| 日本国产一区二区| 91精品啪在线观看国产60岁| 777午夜精品视频在线播放| 在线观看一区二区精品视频| 色妹子一区二区| 91视频免费看| 欧美一级在线免费| 久久久99免费| 久久久久高清精品| 亚洲欧美一区二区三区久本道91| 日韩激情一区二区| 高清成人在线观看| 91黄视频在线| 中文字幕精品在线不卡| 亚洲一区二区偷拍精品| 极品销魂美女一区二区三区| 在线视频一区二区三| 国产精品国产三级国产普通话99| 蜜臀av亚洲一区中文字幕| 91麻豆6部合集magnet| 欧美激情中文不卡| 久久se这里有精品| 欧美一卡2卡三卡4卡5免费| 亚洲国产成人高清精品| 91免费在线看| 中文av一区二区| 成人午夜精品在线| 中文字幕欧美国产| 成人av动漫在线| 国产精品天天看| 色综合色狠狠综合色| 亚洲另类中文字| 91精品国产色综合久久不卡电影| 午夜精彩视频在线观看不卡| 欧美日韩黄视频| 五月开心婷婷久久| 欧美mv日韩mv国产网站| 久久av中文字幕片| 国产精品美女久久久久久久| 91丨国产丨九色丨pron| 婷婷丁香久久五月婷婷| 国产日韩高清在线| 日本一区二区三区高清不卡| 国产精品自在欧美一区| 国产精品久久久久久久久快鸭| 91免费视频观看|