interface{} 与 []interface{}

原文

简介


基于可以对interface{}赋值任何类型的变量,很多人会尝试如下的代码:

var dataSlice []int = foo()
var interfaceSlice []interface{} = dataSlice

很不幸会导致错误:

cannot use dataSlice (type []int) as type []interface { } in assignment

于是问题来了: 为什么可以把任何类型赋值给interface{},却不能把任何类型的切片赋值到[]interface{}?

原因


导致这个问题,主要有两个原因。

  1. []interface{}类型**不是**interface{}类型, 它是一个切片,切片元素的类型恰好是interface{}。但即便这样解释,有人可能仍然觉得前面的用法没毛病。

  2. 真的没毛病?[]interface{}类型变量拥有特定的内存结构,这在编译时就已经决定了。

    每个interface{}占两个字(word),一个字用于存放interface存放的类型,另一个字用于存放实际数据或者是指向数据的指针。于是长度为N的[]interface{}类型切片背后是一个N*2字长的一块数据。

    这与一般的[]MyType类型切片不同,相同长度的[]MyType切片背后的数据块大小为N*sizeof(MyType)字长。

正是这两个原因决定了不能直接将某些[]MyType切片赋值给[]interface{}, 他们背后代表的数据意义不同。

如何使用


实际上这取决于你的需求。

如果你想得到一个元素为任意类型的列表的容器,并且在索引其中元素之前会把它转换为原本的数据类型,可以直接使用interface{}即可。这种方式很通用(非编译时类型安全:猜测原文作者是想表达被转换为interface{}类型的数据,转换回去也必须是原来的类型,否则编译会报错)也很快速。
参考代码:

package main

import (
    "fmt"
)

type Person interface {
    SayHi() interface{} // 如果此方法返回值换成[]interface{} 会报错
}

type Man struct {
    Name string
}

func (m Man) SayHi() interface{} {
    return []byte("Hi, my name is: " + m.Name)
}

type Women struct {
    Name string
}

func (w Women) SayHi() interface{} {
    return []byte("Hi, my name is: " + w.Name)
}

func sayHi(p Person) {
    fmt.Println(string(p.SayHi().([]byte))) // 转换为[]byte类型
}

func main() {
    m := Man{Name: "Bruce"}
    w := Women{
        Name: "Anna",
    }

    sayHi(m)
    sayHi(w)
}

但如果你需要[]interface{},因为你会在将数据类型转换为原本类型之前索引其中元素,或者你使用某种接口类型需要调用它的方法,那就需要逐个copy得到[]interface{}类型的切片:

var dataSlice []int = foo()
var interfaceSlice []interface{} = make([]interface{}, len(dataSlice))
for i, d := range dataSlice {
    interfaceSlice[i] = d
}

你可能感兴趣的:(go)