• Golang中的interface{}类型
  • 发布于 2个月前
  • 150 热度
    0 评论
在本文中,我们将深入探讨interface{}类型,并全面了解它的运作方式。让我们深入探讨。
1. 接口实际上包含两个东西,而不是一个
在运行时,这个过程被称为“装箱”值,你可以将interface{}想象成这样:
type iface struct {
    tab  *itab
    data unsafe.Pointer
}
类型描述符tab是一个包含接口方法集、底层类型以及实现接口的底层类型的方法的小型结构体:
type itab struct {
    inter *interfacetype
    _type *_type
    hash  uint32 // copy of _type.hash. Used for type switches.
    _     [4]byte
    fun   [1]uintptr // variable sized. fun[0]==0 means _type does not implement inter.
}
interface{}变量中指针指向存储值的内存位置。当提取一个interface{}值时,Go会通过访问类型描述符和指针来“解箱”它,然后使用这些信息创建一个适当类型的新变量。这个过程由Go运行时处理。
“你是如何了解这些秘密的呢?”

你可以通过查看Go运行时存储库了解更多信息,该存储库包含了一系列函数和数据结构,这些函数和数据结构被Go运行时系统用于管理Go程序的执行。

2. 空不是空
当一个interface{}变量持有一个nil值时,并不意味着它等于nil。下面是一个例子来说明这一点:
func main() {
    var x interface{}
    var y *int = nil
    x = y

    if x != nil {
        fmt.Println("x != nil") // actual
    } else {
        fmt.Println("x == nil") // expect
    }

    fmt.Println(x)
}

// x != nil
// <nil>
在这个例子中,我将x赋值为nil指针int值,但实际结果是"x != nil",即使当我打印x的值时,它也显示为。这是因为当使用"=="运算符将一个interface{}变量与nil进行比较时,只有当类型描述符和值数据都为nil时才会返回true。

“太荒谬了,我该怎么解决这个问题呢?”
我通常使用反射来解决这个问题,因为我不需要知道底层类型。我还会编写一个实用函数来检查nil值:
func IsAnyNil(x interface{}) bool {
     return reflect.ValueOf(x).IsNil()
}
“所以,interface(int64) = 1 和 interface(int32) = 1 不相等... 是吗?”
正如我们之前讨论的,你是正确的,类型为int64的interface{}中的值1不等于类型为int32的interface{}中的值1。为了确认这一点,我为你运行了一个测试:
func main() {
    var y int32 = 1
    var z int64 = 1

    var x1 interface{} = y
    var x2 interface{} = z

    if x1 == x2 {
        fmt.Println("x1 == x2") // expect
    } else {
        fmt.Println("x1 != x2") // actual
    }
}

// x1 != x2
总结
这就结束了我们的讨论。你现在完全理解了interface{}的工作原理,并可以自信地运用这些知识。
用户评论