learn-tech/专栏/TonyBai·Go语言第一课/29接口:为什么nil接口不等于nil?.md
2024-10-16 06:37:41 +08:00

33 KiB
Raw Blame History

                        因收到Google相关通知网站将会择期关闭。相关通知内容
                        
                        
                        29 接口为什么nil接口不等于nil
                        你好我是Tony Bai。

上一讲我们学习了Go接口的基础知识与设计惯例知道Go接口是构建Go应用骨架的重要元素。从语言设计角度来看Go语言的接口interface和并发concurrency原语是我最喜欢的两类Go语言语法元素。Go语言核心团队的技术负责人Russ Cox也曾说过这样一句话“如果要从Go语言中挑选出一个特性放入其他语言我会选择接口”这句话足以说明接口这一语法特性在这位Go语言大神心目中的地位。

为什么接口在Go中有这么高的地位呢这是因为接口是Go这门静态语言中唯一“动静兼备”的语法特性。而且接口“动静兼备”的特性给Go带来了强大的表达能力但同时也给Go语言初学者带来了不少困惑。要想真正解决这些困惑我们必须深入到Go运行时层面看看Go语言在运行时是如何表示接口类型的。在这一讲中我就带着你一起深入到接口类型的运行时表示层面看看。

好,在解惑之前,我们先来看看接口的静态与动态特性,看看“动静皆备”到底是什么意思。

接口的静态特性与动态特性

接口的静态特性体现在接口类型变量具有静态类型比如var err error中变量err的静态类型为error。拥有静态类型那就意味着编译器会在编译阶段对所有接口类型变量的赋值操作进行类型检查编译器会检查右值的类型是否实现了该接口方法集合中的所有方法。如果不满足就会报错

var err error = 1 // cannot use 1 (type int) as type error in assignment: int does not implement error (missing Error method)

而接口的动态特性,就体现在接口类型变量在运行时还存储了右值的真实类型信息,这个右值的真实类型被称为接口类型变量的动态类型。你看一下下面示例代码:

var err error err = errors.New("error1") fmt.Printf("%T\n", err) // *errors.errorString

我们可以看到这个示例通过errros.New构造了一个错误值赋值给了error接口类型变量err并通过fmt.Printf函数输出接口类型变量err的动态类型为*errors.errorString。

那接口的这种“动静皆备”的特性,又带来了什么好处呢?

首先接口类型变量在程序运行时可以被赋值为不同的动态类型变量每次赋值后接口类型变量中存储的动态类型信息都会发生变化这让Go语言可以像动态语言比如Python那样拥有使用Duck Typing鸭子类型的灵活性。所谓鸭子类型就是指某类型所表现出的特性比如是否可以作为某接口类型的右值不是由其基因比如C++中的父类)决定的,而是由类型所表现出来的行为(比如类型拥有的方法)决定的。

比如下面的例子:

type QuackableAnimal interface { Quack() }

type Duck struct{}

func (Duck) Quack() { println("duck quack!") }

type Dog struct{}

func (Dog) Quack() { println("dog quack!") }

type Bird struct{}

func (Bird) Quack() { println("bird quack!") }

func AnimalQuackInForest(a QuackableAnimal) { a.Quack()
}

func main() {
animals := []QuackableAnimal{new(Duck), new(Dog), new(Bird)} for _, animal := range animals { AnimalQuackInForest(animal) }
}

这个例子中我们用接口类型QuackableAnimal来代表具有“会叫”这一特征的动物而Duck、Bird和Dog类型各自都具有这样的特征于是我们可以将这三个类型的变量赋值给QuackableAnimal接口类型变量a。每次赋值变量a中存储的动态类型信息都不同Quack方法的执行结果将根据变量a中存储的动态类型信息而定。

这里的Duck、Bird、Dog都是“鸭子类型”但它们之间并没有什么联系之所以能作为右值赋值给QuackableAnimal类型变量只是因为他们表现出了QuackableAnimal所要求的特征罢了。

不过与动态语言不同的是Go接口还可以保证“动态特性”使用时的安全性。比如编译器在编译期就可以捕捉到将int类型变量传给QuackableAnimal接口类型变量这样的明显错误决不会让这样的错误遗漏到运行时才被发现。

接口类型的动静特性让我们看到了接口类型的强大但在日常使用过程中很多人都会产生各种困惑其中最经典的一个困惑莫过于“nil的error值不等于nil”了。下面我们来详细看一下。

nil error值 != nil

这里我们直接来看一段改编自GO FAQ中的例子的代码

type MyError struct { error }

var ErrBad = MyError{ error: errors.New("bad things happened"), }

func bad() bool { return false }

func returnsError() error { var p *MyError = nil if bad() { p = &ErrBad } return p }

func main() { err := returnsError() if err != nil { fmt.Printf("error occur: %+v\n", err) return } fmt.Println("ok") }

在这个例子中我们的关注点集中在returnsError这个函数上面。这个函数定义了一个*MyError类型的变量p初值为nil。如果函数bad返回falsereturnsError函数就会直接将p此时p = nil作为返回值返回给调用者之后调用者会将returnsError函数的返回值error接口类型与nil进行比较并根据比较结果做出最终处理。

如果你是一个初学者我猜你的的思路大概是这样的p为nilreturnsError返回p那么main函数中的err就等于nil于是程序输出ok后退出。

但真实的运行结果是什么样的呢?我们来看一下:

error occur:

我们看到示例程序并未如我们前面预期的那样输出ok。程序显然是进入了错误处理分支输出了err的值。那这里就有一个问题了明明returnsError函数返回的p值为nil为什么却满足了if err != nil的条件进入错误处理分支呢

要想弄清楚这个问题,我们需要进一步了解接口类型变量的内部表示。

接口类型变量的内部表示

接口类型“动静兼备”的特性也决定了它的变量的内部表示绝不像一个静态类型变量如int、float64那样简单我们可以在$GOROOT/src/runtime/runtime2.go中找到接口类型变量在运行时的表示

// $GOROOT/src/runtime/runtime2.go type iface struct { tab *itab data unsafe.Pointer }

type eface struct { _type *_type data unsafe.Pointer }

我们看到在运行时层面接口类型变量有两种内部表示iface和eface这两种表示分别用于不同的接口类型变量

eface用于表示没有方法的空接口empty interface类型变量也就是interface{}类型的变量; iface用于表示其余拥有方法的接口interface类型变量。

这两个结构的共同点是它们都有两个指针字段,并且第二个指针字段的功能相同,都是指向当前赋值给该接口类型变量的动态类型变量的值。

那它们的不同点在哪呢就在于eface表示的空接口类型并没有方法列表因此它的第一个指针字段指向一个_type类型结构这个结构为该接口类型变量的动态类型的信息它的定义是这样的

// $GOROOT/src/runtime/type.go

type _type struct { size uintptr ptrdata uintptr // size of memory prefix holding all pointers hash uint32 tflag tflag align uint8 fieldAlign uint8 kind uint8 // function for comparing objects of this type // (ptr to object A, ptr to object B) -> ==? equal func(unsafe.Pointer, unsafe.Pointer) bool // gcdata stores the GC type data for the garbage collector. // If the KindGCProg bit is set in kind, gcdata is a GC program. // Otherwise it is a ptrmask bitmap. See mbitmap.go for details. gcdata *byte str nameOff ptrToThis typeOff }

而iface除了要存储动态类型信息之外还要存储接口本身的信息接口的类型信息、方法列表信息等以及动态类型所实现的方法的信息因此iface的第一个字段指向一个itab类型结构。itab结构的定义如下

// $GOROOT/src/runtime/runtime2.go 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. }

这里我们也可以看到itab结构中的第一个字段inter指向的interfacetype结构存储着这个接口类型自身的信息。你看一下下面这段代码表示的interfacetype类型定义 这个interfacetype结构由类型信息typ、包路径名pkgpath和接口方法集合切片mhdr组成。

// $GOROOT/src/runtime/type.go type interfacetype struct { typ _type pkgpath name mhdr []imethod }

itab结构中的字段_type则存储着这个接口类型变量的动态类型的信息字段fun则是动态类型已实现的接口方法的调用地址数组。

下面我们再结合例子用图片来直观展现eface和iface的结构。首先我们看一个用eface表示的空接口类型变量的例子

type T struct { n int s string }

func main() { var t = T { n: 17, s: "hello, interface", }

var ei interface{} = t // Go运行时使用eface结构表示ei

}

这个例子中的空接口类型变量ei在Go运行时的表示是这样的

我们看到空接口类型的表示较为简单图中上半部分_type字段指向它的动态类型T的类型信息下半部分的data则是指向一个T类型的实例值。

我们再来看一个更复杂的用iface表示非空接口类型变量的例子

type T struct { n int s string }

func (T) M1() {} func (T) M2() {}

type NonEmptyInterface interface { M1() M2() }

func main() { var t = T{ n: 18, s: "hello, interface", } var i NonEmptyInterface = t }

和eface比起来iface的表示稍微复杂些。我也画了一幅表示上面NonEmptyInterface接口类型变量在Go运行时表示的示意图

由上面的这两幅图我们可以看出每个接口类型变量在运行时的表示都是由两部分组成的针对不同接口类型我们可以简化记作eface(_type, data)和iface(tab, data)。

而且虽然eface和iface的第一个字段有所差别但tab和_type可以统一看作是动态类型的类型信息。Go语言中每种类型都会有唯一的_type信息无论是内置原生类型还是自定义类型都有。Go运行时会为程序内的全部类型建立只读的共享_type信息表因此拥有相同动态类型的同类接口类型变量的_type/tab信息是相同的。

而接口类型变量的data部分则是指向一个动态分配的内存空间这个内存空间存储的是赋值给接口类型变量的动态类型变量的值。未显式初始化的接口类型变量的值为nil也就是这个变量的_type/tab和data都为nil。

也就是说我们判断两个接口类型变量是否相同只需要判断_type/tab是否相同以及data指针指向的内存空间所存储的数据值是否相同就可以了。这里要注意不是data指针的值相同噢。

不过通过肉眼去辨别接口类型变量是否相等总是困难一些我们可以引入一些helper函数。借助这些函数我们可以清晰地输出接口类型变量的内部表示这样就可以一目了然地看出两个变量是否相等了。

由于eface和iface是runtime包中的非导出结构体定义我们不能直接在包外使用所以也就无法直接访问到两个结构体中的数据。不过Go语言提供了println预定义函数可以用来输出eface或iface的两个指针字段的值。

在编译阶段编译器会根据要输出的参数的类型将println替换为特定的函数这些函数都定义在$GOROOT/src/runtime/print.go文件中而针对eface和iface类型的打印函数实现如下

// $GOROOT/src/runtime/print.go func printeface(e eface) { print("(", e._type, ",", e.data, ")") }

func printiface(i iface) { print("(", i.tab, ",", i.data, ")") }

我们看到printeface和printiface会输出各自的两个指针字段的值。下面我们就来使用println函数输出各类接口类型变量的内部表示信息并结合输出结果解析接口类型变量的等值比较操作。

第一种nil接口变量

我们前面提过未赋初值的接口类型变量的值为nil这类变量也就是nil接口变量我们来看这类变量的内部表示输出的例子

func printNilInterface() { // nil接口变量 var i interface{} // 空接口类型 var err error // 非空接口类型 println(i) println(err) println("i = nil:", i == nil) println("err = nil:", err == nil) println("i = err:", i == err) }

运行这个函数,输出结果是这样的:

(0x0,0x0) (0x0,0x0) i = nil: true err = nil: true i = err: true

我们看到无论是空接口类型还是非空接口类型变量一旦变量值为nil那么它们内部表示均为(0x0,0x0)也就是类型信息、数据值信息均为空。因此上面的变量i和err等值判断为true。

第二种:空接口类型变量

下面是空接口类型变量的内部表示输出的例子:

func printEmptyInterface() { var eif1 interface{} // 空接口类型 var eif2 interface{} // 空接口类型 var n, m int = 17, 18

  eif1 = n
  eif2 = m

  println("eif1:", eif1)
  println("eif2:", eif2)
  println("eif1 = eif2:", eif1 == eif2) // false

  eif2 = 17
  println("eif1:", eif1)
  println("eif2:", eif2)
  println("eif1 = eif2:", eif1 == eif2) // true

  eif2 = int64(17)
  println("eif1:", eif1)
  println("eif2:", eif2)
  println("eif1 = eif2:", eif1 == eif2) // false

}

这个例子的运行输出结果是这样的:

eif1: (0x10ac580,0xc00007ef48) eif2: (0x10ac580,0xc00007ef40) eif1 = eif2: false eif1: (0x10ac580,0xc00007ef48) eif2: (0x10ac580,0x10eb3d0) eif1 = eif2: true eif1: (0x10ac580,0xc00007ef48) eif2: (0x10ac640,0x10eb3d8) eif1 = eif2: false

我们按顺序分析一下这个输出结果。

首先代码执行到第11行时eif1与eif2已经分别被赋值整型值17与18这样eif1和eif2的动态类型的类型信息是相同的都是0x10ac580但data指针指向的内存块中存储的值不同一个是17一个是18于是eif1不等于eif2。

接着代码执行到第16行的时候eif2已经被重新赋值为17这样eif1和eif2不仅存储的动态类型的类型信息是相同的都是0x10ac580data指针指向的内存块中存储值也相同了都是17于是eif1等于eif2。

然后代码执行到第21行时eif2已经被重新赋值了int64类型的数值17。这样eif1和eif2存储的动态类型的类型信息就变成不同的了一个是int一个是int64即便data指针指向的内存块中存储值是相同的最终eif1与eif2也是不相等的。

从输出结果中我们可以总结一下对于空接口类型变量只有_type和data所指数据内容一致的情况下两个空接口类型变量之间才能划等号。另外Go在创建eface时一般会为data重新分配新内存空间将动态类型变量的值复制到这块内存空间并将data指针指向这块内存空间。因此我们多数情况下看到的data指针值都是不同的。

第三种:非空接口类型变量

这里,我们也直接来看一个非空接口类型变量的内部表示输出的例子:

type T int

func (t T) Error() string { return "bad error" }

func printNonEmptyInterface() { var err1 error // 非空接口类型 var err2 error // 非空接口类型 err1 = (*T)(nil) println("err1:", err1) println("err1 = nil:", err1 == nil)

err1 = T(5)
err2 = T(6)
println("err1:", err1)
println("err2:", err2)
println("err1 = err2:", err1 == err2)

err2 = fmt.Errorf("%d\n", 5)
println("err1:", err1)
println("err2:", err2)
println("err1 = err2:", err1 == err2)

}

这个例子的运行输出结果如下:

err1: (0x10ed120,0x0) err1 = nil: false err1: (0x10ed1a0,0x10eb310) err2: (0x10ed1a0,0x10eb318) err1 = err2: false err1: (0x10ed1a0,0x10eb310) err2: (0x10ed0c0,0xc000010050) err1 = err2: false

我们看到上面示例中每一轮通过println输出的err1和err2的tab和data值要么data值不同要么tab与data值都不同。

和空接口类型变量一样只有tab和data指的数据内容一致的情况下两个非空接口类型变量之间才能划等号。这里我们要注意err1下面的赋值情况

err1 = (*T)(nil)

针对这种赋值println输出的err1是0x10ed120, 0x0也就是非空接口类型变量的类型信息并不为空数据指针为空因此它与nil0x0,0x0之间不能划等号。

现在我们再回到我们开头的那个问题你是不是已经豁然开朗了呢开头的问题中从returnsError返回的error接口类型变量err的数据指针虽然为空但它的类型信息iface.tab并不为空而是*MyError对应的类型信息这样err与nil0x0,0x0相比自然不相等这就是我们开头那个问题的答案解析现在你明白了吗

第四种:空接口类型变量与非空接口类型变量的等值比较

下面是非空接口类型变量和空接口类型变量之间进行比较的例子:

func printEmptyInterfaceAndNonEmptyInterface() { var eif interface{} = T(5) var err error = T(5) println("eif:", eif) println("err:", err) println("eif = err:", eif == err)

err = T(6)
println("eif:", eif)
println("err:", err)
println("eif = err:", eif == err)

}

这个示例的输出结果如下:

eif: (0x10b3b00,0x10eb4d0) err: (0x10ed380,0x10eb4d8) eif = err: true eif: (0x10b3b00,0x10eb4d0) err: (0x10ed380,0x10eb4e0) eif = err: false

你可以看到空接口类型变量和非空接口类型变量内部表示的结构有所不同第一个字段_type vs. tab)两者似乎一定不能相等。但Go在进行等值比较时类型比较使用的是eface的_type和iface的tab._type因此就像我们在这个例子中看到的那样当eif和err都被赋值为T(5)时,两者之间是划等号的。

好了到这里我们已经学完了各类接口类型变量在运行时层的表示。我们可以通过println可以查看这个表示信息从中我们也知道了接口变量只有在类型信息与值信息都一致的情况下才能划等号。

输出接口类型变量内部表示的详细信息

不过println输出的接口类型变量的内部表示信息在一般情况下都是足够的但有些时候又显得过于简略比如在上面最后一个例子中如果仅凭eif: (0x10b3b00,0x10eb4d0)和err: (0x10ed380,0x10eb4d8)的输出,我们是无法想到两个变量是相等的。

那这时如果我们能输出接口类型变量内部表示的详细信息比如tab._type那势必可以取得事半功倍的效果。接下来我们就看看这要怎么做。

前面提到过eface和iface以及组成它们的itab和_type都是runtime包下的非导出结构体我们无法在外部直接引用它们。但我们发现组成eface、iface的类型都是基本数据类型我们完全可以通过“复制代码”的方式将它们拿到runtime包外面来。

不过这里要注意由于runtime中的eface、iface或者它们的组成可能会随着Go版本的变化发生变化因此这个方法不具备跨版本兼容性。也就是说基于Go 1.17版本复制的代码可能仅适用于使用Go 1.17版本编译。这里我们就以Go 1.17版本为例看看:

// dumpinterface.go type eface struct { _type *_type data unsafe.Pointer }

type tflag uint8 type nameOff int32 type typeOff int32

type _type struct { size uintptr ptrdata uintptr // size of memory prefix holding all pointers hash uint32 tflag tflag align uint8 fieldAlign uint8 kind uint8 // function for comparing objects of this type // (ptr to object A, ptr to object B) -> ==? equal func(unsafe.Pointer, unsafe.Pointer) bool // gcdata stores the GC type data for the garbage collector. // If the KindGCProg bit is set in kind, gcdata is a GC program. // Otherwise it is a ptrmask bitmap. See mbitmap.go for details. gcdata *byte str nameOff ptrToThis typeOff }

type iface struct { tab *itab data unsafe.Pointer }

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. }

... ...

const ptrSize = unsafe.Sizeof(uintptr(0))

func dumpEface(i interface{}) { ptrToEface := (*eface)(unsafe.Pointer(&i)) fmt.Printf("eface: %+v\n", *ptrToEface)

if ptrToEface._type != nil {
    // dump _type info
    fmt.Printf("\t _type: %+v\n", *(ptrToEface._type))
}

if ptrToEface.data != nil {
    // dump data
    switch i.(type) {
    case int:
        dumpInt(ptrToEface.data)
    case float64:
        dumpFloat64(ptrToEface.data)
    case T:
        dumpT(ptrToEface.data)

    // other cases ... ...
    default:
        fmt.Printf("\t unsupported data type\n")
    }
}
fmt.Printf("\n")

}

func dumpItabOfIface(ptrToIface unsafe.Pointer) { p := (*iface)(ptrToIface) fmt.Printf("iface: %+v\n", *p)

if p.tab != nil {
    // dump itab
    fmt.Printf("\t itab: %+v\n", *(p.tab))
    // dump inter in itab
    fmt.Printf("\t\t inter: %+v\n", *(p.tab.inter))

    // dump _type in itab
    fmt.Printf("\t\t _type: %+v\n", *(p.tab._type))

    // dump fun in tab
    funPtr := unsafe.Pointer(&(p.tab.fun))
    fmt.Printf("\t\t fun: [")
    for i := 0; i < len((*(p.tab.inter)).mhdr); i++ {
        tp := (*uintptr)(unsafe.Pointer(uintptr(funPtr) + uintptr(i)*ptrSize))
        fmt.Printf("0x%x(%d),", *tp, *tp)
    }
    fmt.Printf("]\n")
}

}

func dumpDataOfIface(i interface{}) { // this is a trick as the data part of eface and iface are same ptrToEface := (*eface)(unsafe.Pointer(&i))

if ptrToEface.data != nil {
    // dump data
    switch i.(type) {
    case int:
        dumpInt(ptrToEface.data)
    case float64:
        dumpFloat64(ptrToEface.data)
    case T:
        dumpT(ptrToEface.data)

    // other cases ... ...

    default:
        fmt.Printf("\t unsupported data type\n")
    }
}
fmt.Printf("\n")

}

func dumpT(dataOfIface unsafe.Pointer) { var p *T = (*T)(dataOfIface) fmt.Printf("\t data: %+v\n", *p) } ... ...

这里我挑选了关键部分省略了部分代码。上面这个dumpinterface.go中提供了三个主要函数:

dumpEface: 用于输出空接口类型变量的内部表示信息; dumpItabOfIface: 用于输出非空接口类型变量的tab字段信息 dumpDataOfIface: 用于输出非空接口类型变量的data字段信息

我们利用这三个函数来输出一下前面printEmptyInterfaceAndNonEmptyInterface函数中的接口类型变量的信息

package main

import "unsafe"

type T int

func (t T) Error() string { return "bad error" }

func main() { var eif interface{} = T(5) var err error = T(5) println("eif:", eif) println("err:", err) println("eif = err:", eif == err)

dumpEface(eif)
dumpItabOfIface(unsafe.Pointer(&err))
dumpDataOfIface(err)

}

运行这个示例代码,我们得到了这个输出结果:

eif: (0x10b38c0,0x10e9b30) err: (0x10eb690,0x10e9b30) eif = err: true eface: {_type:0x10b38c0 data:0x10e9b30} _type: {size:8 ptrdata:0 hash:1156555957 tflag:15 align:8 fieldAlign:8 kind:2 equal:0x10032e0 gcdata:0x10e9a60 str:4946 ptrToThis:58496} data: bad error

iface: {tab:0x10eb690 data:0x10e9b30} itab: {inter:0x10b5e20 _type:0x10b38c0 hash:1156555957 _:[0 0 0 0] fun:[17454976]} inter: {typ:{size:16 ptrdata:16 hash:235953867 tflag:7 align:8 fieldAlign:8 kind:20 equal:0x10034c0 gcdata:0x10d2418 str:3666 ptrToThis:26848} pkgpath:{bytes:} mhdr:[{name:2592 ityp:43520}]} _type: {size:8 ptrdata:0 hash:1156555957 tflag:15 align:8 fieldAlign:8 kind:2 equal:0x10032e0 gcdata:0x10e9a60 str:4946 ptrToThis:58496} fun: [0x10a5780(17454976),] data: bad error

从输出结果中我们看到eif的_type0x10b38c0与err的tab._type0x10b38c0是一致的data指针所指内容“bad error”也是一致的因此eif == err表达式的结果为true。

再次强调一遍上面这个实现可能仅在Go 1.17版本上测试通过并且在输出iface或eface的data部分内容时只列出了int、float64和T类型的数据读取实现没有列出全部类型的实现你可以根据自己的需要实现其余数据类型。dumpinterface.go的完整代码你可以在这里找到。

我们现在已经知道了接口类型有着复杂的内部结构所以我们将一个类型变量值赋值给一个接口类型变量值的过程肯定不会像var i int = 5那么简单那么接口类型变量赋值的过程是怎样的呢其实接口类型变量赋值是一个“装箱”的过程。

接口类型的装箱boxing原理

装箱boxing是编程语言领域的一个基础概念一般是指把一个值类型转换成引用类型比如在支持装箱概念的Java语言中将一个int变量转换成Integer对象就是一个装箱操作。

在Go语言中将任意类型赋值给一个接口类型变量也是装箱操作。有了前面对接口类型变量内部表示的学习我们知道接口类型的装箱实际就是创建一个eface或iface的过程。接下来我们就来简要描述一下这个过程也就是接口类型的装箱原理。

我们基于下面这个例子中的接口装箱操作来说明:

// interface_internal.go

type T struct { n int s string }

func (T) M1() {} func (T) M2() {}

type NonEmptyInterface interface { M1() M2() }

func main() { var t = T{ n: 17, s: "hello, interface", } var ei interface{} ei = t

  var i NonEmptyInterface
  i = t
  fmt.Println(ei)
  fmt.Println(i)

}

这个例子中对ei和i两个接口类型变量的赋值都会触发装箱操作要想知道Go在背后做了些什么我们需要“下沉”一层也就是要输出上面Go代码对应的汇编代码

$go tool compile -S interface_internal.go > interface_internal.s

对应ei = t一行的汇编如下

0x0026 00038 (interface_internal.go:24) MOVQ    $17, ""..autotmp_15+104(SP)
0x002f 00047 (interface_internal.go:24) LEAQ    go.string."hello, interface"(SB), CX
0x0036 00054 (interface_internal.go:24) MOVQ    CX, ""..autotmp_15+112(SP)
0x003b 00059 (interface_internal.go:24) MOVQ    $16, ""..autotmp_15+120(SP)
0x0044 00068 (interface_internal.go:24) LEAQ    type."".T(SB), AX
0x004b 00075 (interface_internal.go:24) LEAQ    ""..autotmp_15+104(SP), BX
0x0050 00080 (interface_internal.go:24) PCDATA  $1, $0
0x0050 00080 (interface_internal.go:24) CALL    runtime.convT2E(SB)

对应i = t一行的汇编如下

0x005f 00095 (interface_internal.go:27) MOVQ    $17, ""..autotmp_15+104(SP)
0x0068 00104 (interface_internal.go:27) LEAQ    go.string."hello, interface"(SB), CX
0x006f 00111 (interface_internal.go:27) MOVQ    CX, ""..autotmp_15+112(SP)
0x0074 00116 (interface_internal.go:27) MOVQ    $16, ""..autotmp_15+120(SP)
0x007d 00125 (interface_internal.go:27) LEAQ    go.itab."".T,"".NonEmptyInterface(SB), AX
0x0084 00132 (interface_internal.go:27) LEAQ    ""..autotmp_15+104(SP), BX
0x0089 00137 (interface_internal.go:27) PCDATA  $1, $1
0x0089 00137 (interface_internal.go:27) CALL    runtime.convT2I(SB)

在将动态类型变量赋值给接口类型变量语句对应的汇编代码中我们看到了convT2E和convT2I两个runtime包的函数。这两个函数的实现位于$GOROOT/src/runtime/iface.go中

// $GOROOT/src/runtime/iface.go

func convT2E(t *_type, elem unsafe.Pointer) (e eface) { if raceenabled { raceReadObjectPC(t, elem, getcallerpc(), funcPC(convT2E)) } if msanenabled { msanread(elem, t.size) } x := mallocgc(t.size, t, true) typedmemmove(t, x, elem) e._type = t e.data = x return }

func convT2I(tab *itab, elem unsafe.Pointer) (i iface) { t := tab._type if raceenabled { raceReadObjectPC(t, elem, getcallerpc(), funcPC(convT2I)) } if msanenabled { msanread(elem, t.size) } x := mallocgc(t.size, t, true) typedmemmove(t, x, elem) i.tab = tab i.data = x return }

convT2E用于将任意类型转换为一个efaceconvT2I用于将任意类型转换为一个iface。两个函数的实现逻辑相似主要思路就是根据传入的类型信息convT2E的_type和convT2I的tab._type分配一块内存空间并将elem指向的数据拷贝到这块内存空间中最后传入的类型信息作为返回值结构中的类型信息返回值结构中的数据指针data指向新分配的那块内存空间。

由此我们也可以看出,经过装箱后,箱内的数据,也就是存放在新分配的内存空间中的数据与原变量便无瓜葛了,比如下面这个例子:

func main() { var n int = 61 var ei interface{} = n n = 62 // n的值已经改变 fmt.Println("data in box:", ei) // 输出仍是61 }

那么convT2E和convT2I函数的类型信息是从何而来的呢

其实这些都依赖Go编译器的工作。编译器知道每个要转换为接口类型变量toType和动态类型变量的类型fromType它会根据这一对类型选择适当的convT2X函数并在生成代码时使用选出的convT2X函数参与装箱操作。

不过装箱是一个有性能损耗的操作因此Go也在不断对装箱操作进行优化包括对常见类型如整型、字符串、切片等提供系列快速转换函数

// $GOROOT/src/runtime/iface.go func convT16(val any) unsafe.Pointer // val must be uint16-like func convT32(val any) unsafe.Pointer // val must be uint32-like func convT64(val any) unsafe.Pointer // val must be uint64-like func convTstring(val any) unsafe.Pointer // val must be a string func convTslice(val any) unsafe.Pointer // val must be a slice

这些函数去除了typedmemmove操作增加了零值快速返回等特性。

同时Go建立了staticuint64s区域对255以内的小整数值进行装箱操作时不再分配新内存而是利用staticuint64s区域的内存空间下面是staticuint64s的定义

// $GOROOT/src/runtime/iface.go // staticuint64s is used to avoid allocating in convTx for small integer values. var staticuint64s = [...]uint64{ 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0f, ... ... }

小结

好了,今天的课讲到这里就结束了,现在我们一起来回顾一下吧。

接口类型作为参与构建Go应用骨架的重要参与者在Go语言中有着很高的地位。它这个地位的取得离不开它拥有的“动静兼备”的语法特性。Go接口的动态特性让Go拥有与动态语言相近的灵活性而静态特性又在编译阶段保证了这种灵活性的安全。

要更好地理解Go接口的这两种特性我们需要深入到Go接口在运行时的表示层面上去。接口类型变量在运行时表示为eface和ifaceeface用于表示空接口类型变量iface用于表示非空接口类型变量。只有两个接口类型变量的类型信息eface._type/iface.tab._type相同且数据指针eface.data/iface.data所指数据相同时两个接口类型变量才是相等的。

我们可以通过println输出接口类型变量的两部分指针变量的值。而且通过拷贝runtime包eface和iface相关类型源码我们还可以自定义输出eface/iface详尽信息的函数不过要注意的是由于runtime层代码的演进这个函数可能不具备在Go版本间的移植性。

最后接口类型变量的赋值本质上是一种装箱操作装箱操作是由Go编译器和运行时共同完成的有一定的性能开销对于性能敏感的系统来说我们应该尽量避免或减少这类装箱操作。

思考题

像nil error值 != nil那个例子中的“坑”你在日常编码时有遇到过吗可以和我们分享一下吗另外我们这节课中的这个例子如何修改才能让它按我们最初的预期结果输出呢

欢迎在留言区分享你的经验和想法。也欢迎你把这节课分享给更多对Go接口感兴趣的朋友。我是Tony Bai我们下节课见。