GoLang nil与interface的空指针深入分析
作者:alwaysrun 发布时间:2024-02-18 01:58:50
nil
Go中,每个指针都有2个基本信息,指针的类型和指针的值(type,value);当执行==
时,需要比较类型与值(只有类型与值都相等时,才会相等)。
nil并不是Go语言的关键字或者保留字,而是一个预定义好的标识符:
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,除了不能索引外,其他的操作都正常;当append元素时,slice会自动进行扩容。
slice是一个简单的结构体,包含(长度、容量、指向数组的指针);当slice为nil时,长度、容量都为0,指针为空。
map
一个nil的map,是一个真正的空指针,除len与for-range外,其他操作不能正常使用。
非nil的map,是一个指向内部HashMap的指针;空map(map[string]int{}
)与为nil的map是不同的,空map只是没有内容,可在上面做任何的map操作。
interface
interface底层由两部分组成(参见《golang反射简介》),一个是类型,一个值,也就是类似于:(Type, Value)。只有当类型和值都是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参数,其nil属性不变;若是普通指针,则传递给interface参数时,都为非空(!=nil);
指针是否为空
那如何判定interface里面的动态值是否空?此时需要借助反射reflect来实现:
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
对于非指针类型,在反射后调用IsNil时会抛出异常。其实现:
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


猜你喜欢
- Python工程师面试必备25条Python知识点:1.到底什么是Python?你可以在回答中与其他技术进行对比下面是一些关键点:Pytho
- 主键自增MySQL 提供了主键自增机制 AUTO_INCREMENT. 对主键使用, 保证了主键的唯一性.注意:自增长必须与主键字段配合使用
- 详解python中的文件与目录操作一 获得当前路径1、代码1>>>import os>>>print(&
- 一.re模块的查找方法:1.findall 匹配所有每一项都是列表中的一个元素import reret = re.f
- 在大三的时候,一直就想搭建属于自己的一个博客,但由于各种原因,最终都不了了之,恰好最近比较有空,于是就自己参照网上的教程,搭建了属于自己的博
- 你家中的CD、VCD一定很多吧?是不是常遇到为找一张CD把一抽屉的碟子翻得乱七八糟的情况,你一定没少受埋怨——你不想整理它们一下?如:影片是
- 前言如果你和我一样偶尔看看股票,看看自己关注的股票是涨了还是跌了,或者想快速获取到想看的头条新闻,我们不必把过多的注意力放在去寻找上面,我们
- SQLite支持多种编程语言的开发调用:C, C++ , PHP, Perl, Java, C# ,Python, Ruby等。本篇先介绍P
- 后来查了一些相关资料,发现在IE 中通过 window.location.href 或者是 是无法获取HTTP_REFERER, 真是搞不懂
- 一、什么是组件组件 (Component) 是 Vue.js 最强大的功能之一。组件可以扩展 HTML 元素,封装可重用的代码。二、组件用法
- 本文主要研究的是python将字典内容存入mysql,分享了实现代码,具体介绍如下。1.背景项目需要,用python实现了将字典内容存入本地
- 先看看 allure 命令的帮助文档cmd 敲allure -hallure 命令的语法格式allure [options] [comman
- 1、pyqtgraph库数据可视化效果还不错,特别是窗体程序中图像交互性较好;安装也很方便,用 pip 安装。2、在Python中新建一个
- 用户登录验证脚本,Chkpwd.asp<% '=======用户登录验证脚本======= '如果尚未定义Passed
- 本文实例讲述了JavaScript实现弹出DIV层同时页面背景渐变成半透明效果。分享给大家供大家参考,具体如下:<!DOCTYPE h
- SQL Server 的全文搜索(Full-Text Search)是基于分词的文本检索功能,依赖于全文索引。全文索引不同于传统的平衡树(B
- 本文实例讲述了pytorch制作自己的LMDB数据操作。分享给大家供大家参考,具体如下:前言记录下pytorch里如何使用lmdb的code
- 前言无论是单机锁还是分布式锁,原理都是基于共享的数据,判断当前操作的行为。对于单机则是共享RAM内存,对于集群则可以借助Redis,ZK,D
- Object 类型的对象虽然有 toString 方法,但结果却是 [Object Object] 让人没法理解的字符。比如简单的对象:{n
- random模块用于生成随机数,下面看看模块中一些常用函数的用法:from numpy import randomnumpy.random.