go 语言中的 nil

2022-01-24  本文已影响0人  ag4kd

重点

接口在底层的实现有两个部分:type 和 data。


在源码中,显式地将 nil 赋值给接口时,接口的 type 和 data 都将为 nil。此时,接口与 nil 值判断是相等的。但如果将一个带有类型的 nil 赋值给接口时,只有 data 为 nil,而 type 不为 nil,此时,接口与 nil 判断将不相等。


nil 在 Go语言中只能被赋值给指针和接口。


因此,go 语言中的 nil,是不一定相等,因为接口的 nil 和类型的 nil 在数据结构上一样,但是值不一样.

接口与 nil 不相等

下面代码使用 MyImplement() 实现 fmt 包中的 Stringer 接口,这个接口的定义如下:

type Stringer interface {
    String() string
}

在 GetStringer() 函数中将返回这个接口。通过 *MyImplement 指针变量置为 nil 提供 GetStringer 的返回值。在 main() 中,判断 GetStringer 与 nil 是否相等,代码如下:

package main
import "fmt"
// 定义一个结构体
type MyImplement struct{}
// 实现fmt.Stringer的String方法
func (m *MyImplement) String() string {
    return "hi"
}
// 在函数中返回fmt.Stringer接口
func GetStringer() fmt.Stringer {
    // 赋nil,这句话只是说明 data 为 nil,但是 type 带有 *MyImplement 信息
    var s *MyImplement = nil
    // 返回变量
    //s 变量此时被 fmt.Stringer 接口包装后,实际类型为 *MyImplement,值为 nil 的接口
    return s
}
func main() {
    // 判断返回值是否为nil
    if GetStringer() == nil {
        fmt.Println("GetStringer() == nil")
    } else {
     //使用 GetStringer() 的返回值与 nil 判断时,虽然接口里的 value 为 nil,但 type 带有 *MyImplement 信息,使用 == 判断相等时,依然不为 nil。
        fmt.Println("GetStringer() != nil")
    }
}

换句话说,go 语言中判断相等,需要 typedata同时相等才行。

发现 nil 类型值返回时直接返回 nil

为了避免这类误判的问题,可以在函数返回时,发现带有 nil 的指针时直接返回 nil,代码如下:

func GetStringer() fmt.Stringer {
    var s *MyImplement = nil
    if s == nil {
        return nil
    }
    return s
}
上一篇 下一篇

猜你喜欢

热点阅读