GoLang nil與interface的空指針深入分析
nil
Go中,每個(gè)指針都有2個(gè)基本信息,指針的類型和指針的值(type,value);當(dāng)執(zhí)行==時(shí),需要比較類型與值(只有類型與值都相等時(shí),才會(huì)相等)。
nil并不是Go語(yǔ)言的關(guān)鍵字或者保留字,而是一個(gè)預(yù)定義好的標(biāo)識(shí)符:
- nil之間不能比較:
nil==nil是不允許的,會(huì)拋出operator == not defined on untyped nil異常; - 不同類型的nil之間不能互相比較:如切片的nil,不能與map的nil做比較;
- nil是
map、slice、pointer、channel、func、interface的零值; - 不同類型nil值占用空間可能大小不同;
在64位機(jī)器上運(yùn)行時(shí)nil的大?。?/p>
func main() {
var p *struct{}
fmt.Println(unsafe.Sizeof(p), p == nil) // 8
var s []int
fmt.Println(unsafe.Sizeof(s), s == nil) // 24
var m map[int]bool
fmt.Println(unsafe.Sizeof(m), m == nil) // 8
var c chan string
fmt.Println(unsafe.Sizeof(c), c == nil) // 8
var f func()
fmt.Println(unsafe.Sizeof(f), f == nil) // 8
var i interface{}
fmt.Println(unsafe.Sizeof(i), i == nil) // 16
}
slice
一個(gè)nil的slice,除了不能索引外,其他的操作都正常;當(dāng)append元素時(shí),slice會(huì)自動(dòng)進(jìn)行擴(kuò)容。
slice是一個(gè)簡(jiǎn)單的結(jié)構(gòu)體,包含(長(zhǎng)度、容量、指向數(shù)組的指針);當(dāng)slice為nil時(shí),長(zhǎng)度、容量都為0,指針為空。


map
一個(gè)nil的map,是一個(gè)真正的空指針,除len與for-range外,其他操作不能正常使用。
非nil的map,是一個(gè)指向內(nèi)部HashMap的指針;空map(map[string]int{})與為nil的map是不同的,空map只是沒(méi)有內(nèi)容,可在上面做任何的map操作。
interface
interface底層由兩部分組成(參見(jiàn)《golang反射簡(jiǎn)介》),一個(gè)是類型,一個(gè)值,也就是類似于:(Type, Value)。只有當(dāng)類型和值都是nil的時(shí)候,才等于nil:
func inFun(v interface{}) {
fmt.Println("fun-interface:", v == nil)
}
func main() {
var a interface{}
var b []string
var c string
fmt.Println(a == nil)
inFun(a) // true
fmt.Println(b == nil)
inFun(b) // false
//fmt.Println(c == nil) // can not compare with nil
inFun(c) // false
}
// true
// fun-interface: true
// true
// fun-interface: false
// fun-interface: false本身是interface時(shí),傳遞interface參數(shù),其nil屬性不變;若是普通指針,則傳遞給interface參數(shù)時(shí),都為非空(!=nil);
指針是否為空
那如何判定interface里面的動(dòng)態(tài)值是否空?此時(shí)需要借助反射reflect來(lái)實(shí)現(xiàn):
func nilCheck(v interface{}) {
defer func() {
if err := recover(); err != nil {
fmt.Println("panic:", err)
}
}()
if v == nil {
fmt.Println("nilCheck: interface is nil")
return
}
vi := reflect.ValueOf(v)
fmt.Println("nilCheck:", vi.IsNil())
}
func main() {
var a interface{}
var b []string
var c string
nilCheck(a)
nilCheck(b)
nilCheck(c)
}
// nilCheck: interface is nil
// nilCheck: true
// panic: reflect: call of reflect.Value.IsNil on string Value
對(duì)于非指針類型,在反射后調(diào)用IsNil時(shí)會(huì)拋出異常。其實(shí)現(xiàn):
func (v Value) IsNil() bool {
k := v.kind()
switch k {
case Chan, Func, Map, Pointer, UnsafePointer:
if v.flag&flagMethod != 0 {
return false
}
ptr := v.ptr
if v.flag&flagIndir != 0 {
ptr = *(*unsafe.Pointer)(ptr)
}
return ptr == nil
case Interface, Slice:
// Both interface and slice are nil if first word is 0.
// Both are always bigger than a word; assume flagIndir.
return *(*unsafe.Pointer)(v.ptr) == nil
}
panic(&ValueError{"reflect.Value.IsNil", v.kind()})
}
到此這篇關(guān)于GoLang nil與interface的空指針深入分析的文章就介紹到這了,更多相關(guān)GoLang nil內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
web項(xiàng)目中g(shù)olang性能監(jiān)控解析
這篇文章主要為大家介紹了web項(xiàng)目中g(shù)olang性能監(jiān)控詳細(xì)的解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步早日升職加薪2022-04-04
詳解Golang中文件系統(tǒng)事件監(jiān)聽(tīng)
文件系統(tǒng)事件是指文件系統(tǒng)相關(guān)的各種操作和狀態(tài)變化,當(dāng)一個(gè)應(yīng)用層的進(jìn)程操作文件或目錄時(shí),會(huì)觸發(fā)system call,內(nèi)核的notification子系統(tǒng)可以守在那里,把該進(jìn)程對(duì)文件的操作上報(bào)給應(yīng)用層的監(jiān)聽(tīng)進(jìn)程,這篇文章主要介紹了Golang之文件系統(tǒng)事件監(jiān)聽(tīng),需要的朋友可以參考下2024-01-01
基于Go和PHP語(yǔ)言實(shí)現(xiàn)爬樓梯算法的思路詳解
這篇文章主要介紹了Go和PHP 實(shí)現(xiàn)爬樓梯算法,本文通過(guò)動(dòng)態(tài)規(guī)劃和斐波那契數(shù)列兩種解決思路給大家講解的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-05-05
Go語(yǔ)言實(shí)現(xiàn)23種設(shè)計(jì)模式的使用
設(shè)計(jì)模式是軟件工程中各種常見(jiàn)問(wèn)題的經(jīng)典解決方案,,本文主要介紹了Go語(yǔ)言實(shí)現(xiàn)23種設(shè)計(jì)模式的使用,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-05-05
Go語(yǔ)言sync.Map詳解及使用場(chǎng)景
Go語(yǔ)言中的sync.Map是一個(gè)高效的并發(fā)安全映射結(jié)構(gòu),適用于高并發(fā)讀多寫(xiě)少的場(chǎng)景,它通過(guò)讀寫(xiě)分離、無(wú)鎖讀取路徑、寫(xiě)入時(shí)的鎖保護(hù)等機(jī)制,提高了讀取性能并減少了鎖競(jìng)爭(zhēng),sync.Map不需要手動(dòng)管理鎖,降低了編程復(fù)雜性,適合需要簡(jiǎn)單并發(fā)訪問(wèn)的場(chǎng)合2024-10-10
Golang基礎(chǔ)學(xué)習(xí)之map的示例詳解
哈希表是常見(jiàn)的數(shù)據(jù)結(jié)構(gòu),有的語(yǔ)言會(huì)將哈希稱作字典或者映射,在Go中,哈希就是常見(jiàn)的數(shù)據(jù)類型map,本文就來(lái)聊聊Golang中map的相關(guān)知識(shí)吧2023-03-03
Go語(yǔ)言定時(shí)器Timer和Ticker的使用與區(qū)別
在Go語(yǔ)言中內(nèi)置的有兩個(gè)定時(shí)器,Timer和Ticker,本文主要介紹了Go語(yǔ)言定時(shí)器Timer和Ticker的使用與區(qū)別,具有一定的參考價(jià)值,感興趣的可以了解一下2024-07-07

