GoLang nil与interface的空指针深入分析

 更新时间:2022年12月23日 09:54:01   作者:alwaysrun  
Go语言中任何类型在未初始化时都对应一个零值:布尔类型是false,整型是0,字符串是"",而指针、函数、interface、slice、channel和map的零值都是nil

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()})
}

到此这篇关于GoLang nil与interface的空指针深入分析的文章就介绍到这了,更多相关GoLang nil内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Go工具链之代码测试神器go test详解

    Go工具链之代码测试神器go test详解

    这篇文章主要给大家介绍Go 工具链go test,go test 是 Go 工具链中的一个命令,用于编译和运行按照要求编写的 Golang 测试代码,并生成测试报告,感兴趣的同学跟着小编一起来看看本文吧
    2023-07-07
  • 一文带你深入了解Go语言中的事务

    一文带你深入了解Go语言中的事务

    事务中止时,你结束事务了吗?在开发时有可能就会犯这样的错误,其问题就是你在提交事务时,如果中间有其他业务就取消操作,那么事务也关闭了吗?本文就来详细讲讲
    2023-04-04
  • Go语言基础知识点介绍

    Go语言基础知识点介绍

    在本篇文章里小编给大家整理的是一篇关于Go语言基础知识点介绍内容,有兴趣的朋友们可以跟着学习参考下。
    2021-07-07
  • golang编程开发使用sort排序示例详解

    golang编程开发使用sort排序示例详解

    这篇文章主要为大家介绍了go语言编程开发使用sort来排序示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步早日升职加薪
    2021-11-11
  • gin正确多次读取http request body内容实现详解

    gin正确多次读取http request body内容实现详解

    这篇文章主要为大家介绍了gin正确多次读取http request body内容实现详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-01-01
  • go中的protobuf和grpc使用教程

    go中的protobuf和grpc使用教程

    gRPC 是 Google 公司基于 Protobuf 开发的跨语言的开源 RPC 框架,这篇文章主要介绍了go中的protobuf和grpc使用教程,需要的朋友可以参考下
    2024-08-08
  • go语言中数据接口set集合的实现

    go语言中数据接口set集合的实现

    set集合是一种常见的数据结构,它代表了一个唯一元素的集合,本文主要介绍了set的基本特性,包括唯一性、无序性、可变性和集合运算,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2024-10-10
  • Go语言题解LeetCode1260二维网格迁移示例详解

    Go语言题解LeetCode1260二维网格迁移示例详解

    这篇文章主要为大家介绍了Go语言题解LeetCode1260二维网格迁移示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-01-01
  • Go语言原子操作atomic的使用

    Go语言原子操作atomic的使用

    本文介绍了Go语言原子操作的使用方法,原子操作是一种无锁的技术,可通过CPU指令实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2024-10-10
  • go语言中[]*int和*[]int的具体使用

    go语言中[]*int和*[]int的具体使用

    本文主要介绍了go语言中[]*int和*[]int的具体使用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-04-04

最新评论