網(wǎng)站首頁 編程語言 正文
nil
Go中,每個指針都有2個基本信息,指針的類型和指針的值(type,value);當(dāng)執(zhí)行==
時,需要比較類型與值(只有類型與值都相等時,才會相等)。
nil并不是Go語言的關(guān)鍵字或者保留字,而是一個預(yù)定義好的標(biāo)識符:
- nil之間不能比較:
nil==nil
是不允許的,會拋出operator == not defined on untyped nil
異常; - 不同類型的nil之間不能互相比較:如切片的nil,不能與map的nil做比較;
- nil是
map、slice、pointer、channel、func、interface
的零值; - 不同類型nil值占用空間可能大小不同;
在64位機器上運行時nil的大小:
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
一個nil的slice,除了不能索引外,其他的操作都正常;當(dāng)append元素時,slice會自動進行擴容。
slice是一個簡單的結(jié)構(gòu)體,包含(長度、容量、指向數(shù)組的指針);當(dāng)slice為nil時,長度、容量都為0,指針為空。
map
一個nil的map,是一個真正的空指針,除len與for-range外,其他操作不能正常使用。
非nil的map,是一個指向內(nèi)部HashMap的指針;空map(map[string]int{}
)與為nil的map是不同的,空map只是沒有內(nèi)容,可在上面做任何的map操作。
interface
interface底層由兩部分組成(參見《golang反射簡介》),一個是類型,一個值,也就是類似于:(Type, Value)。只有當(dāng)類型和值都是nil的時候,才等于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時,傳遞interface參數(shù),其nil屬性不變;若是普通指針,則傳遞給interface參數(shù)時,都為非空(!=nil);
指針是否為空
那如何判定interface里面的動態(tài)值是否空?此時需要借助反射reflect來實現(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
對于非指針類型,在反射后調(diào)用IsNil時會拋出異常。其實現(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()}) }
原文鏈接:https://blog.csdn.net/alwaysrun/article/details/127598335
相關(guān)推薦
- 2022-06-01 Androidstudio調(diào)用攝像頭拍照并保存照片_Android
- 2024-01-28 spring自動配置的原理
- 2022-06-12 C#異步編程的三種模式_C#教程
- 2022-05-16 .NET中的HashSet及原理解析_實用技巧
- 2022-11-17 Android自定義一個view?ViewRootImpl繪制流程示例_Android
- 2022-04-08 Python編程-封裝,繼承與多態(tài)_python
- 2022-10-03 Golang實現(xiàn)HTTP編程請求和響應(yīng)_Golang
- 2022-06-01 Python?中enum的使用方法總結(jié)_python
- 最近更新
-
- window11 系統(tǒng)安裝 yarn
- 超詳細win安裝深度學(xué)習(xí)環(huán)境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎(chǔ)操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區(qū)別,Jav
- spring @retryable不生效的一種
- Spring Security之認(rèn)證信息的處理
- Spring Security之認(rèn)證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權(quán)
- redisson分布式鎖中waittime的設(shè)
- maven:解決release錯誤:Artif
- restTemplate使用總結(jié)
- Spring Security之安全異常處理
- MybatisPlus優(yōu)雅實現(xiàn)加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務(wù)發(fā)現(xiàn)-Nac
- Spring Security之基于HttpR
- Redis 底層數(shù)據(jù)結(jié)構(gòu)-簡單動態(tài)字符串(SD
- arthas操作spring被代理目標(biāo)對象命令
- Spring中的單例模式應(yīng)用詳解
- 聊聊消息隊列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支