您可以参考文章“ 如何在Go中使用接口 ”(基于“ Russ Cox对接口的描述
”):
什么 是 接口?
接口是两件事:
- 这是一组方法,
- 但这也是一种
该
interface{}类型的 空接口 是没有方法的接口。由于没有Implements关键字,所有类型都至少实现了零个方法,并且自动完成了一个接口的满足,所以 所有类型都满足空接口 。
这意味着,如果编写一个将interface{}值作为参数的函数,则 可以为该函数提供任何value 。
(这
Msg代表您的问题:任何值)
func DoSomething(v interface{}) { // ...}这是令人困惑的地方:
在内部
DoSomething功能, 是什么v的类型?导致地鼠的人相信“
v任何类型”,但这是错误的。v不是任何类型; 它是interface{}类型的。当将值传递给
DoSomething函数时,Go运行时将执行 类型转换 (如有必要),
并将该值转换为interface{}value。
所有值在运行时都只有一种类型,而v一个静态类型是interface{}。接口值由两个数据字构成 :
- 一个单词用于指向该值的基础类型的方法表,
- 换句话讲,就是指向该值所保存的实际数据。
附录:这是Russ的关于接口结构的文章非常完整:
type Stringer interface { String() string}接口值表示为两字对,它给出了指向有关接口中存储的类型的信息的指针以及指向相关数据的指针。
将b分配给Stringer类型的接口值可设置接口值的两个字。
接口值中的第一个单词指向我所谓的接口表或itable (发音为i-table;在运行时源中,C实现名称为Itab)。
它以有关所涉及类型的一些元数据开始,然后成为函数指针的列表。
请注意,itable对应于接口类型,而不是动态类型 。就我们的示例而言,可保存
Stringer类型Binary的可列出性列出了用于满足Stringer的方法,而这些方法只是String:Binary的其他方法(Get)在中没有出现itable。接口值中的第二个单词指向实际数据 ,在这种情况下为的副本
b。
该赋值var s Stringer = b进行复制b而不是指向,b其原因与var c uint64 =b进行复制的原因相同:如果b以后进行更改,s并且c应该具有原始值,而不是新值。
存储在接口中的值可能任意大,但只有一个字专用于在接口结构中保存该值,因此该分配在堆上分配了一块内存,并将指针记录在一个字槽中。



