2 回答
TA贡献1876条经验 获得超5个赞
Slice 基本上只是对底层数组、起始指针、长度和容量的引用。因此,如果可能的话,请考虑以下事项:
sliceOfStrings := []string{"one", "two", "three"}
// prints ONE TWO THREE
for i := range sliceOfStrings {
fmt.Println(strings.ToUpper(sliceOfStrings[i]))
}
// imagine this is possible
var sliceOfInterface = []interface{}(sliceOfStrings)
// since it's array of interface{} now - we can do anything
// let's put integer into the first position
sliceOfInterface[0] = 1
// sliceOfStrings still points to the same array, and now "one" is replaced by 1
fmt.Println(strings.ToUpper(sliceOfStrings[0])) // BANG!
此问题存在于 Java 和 C# 中。在实践中它很少发生,但仍然发生。鉴于在 Go 中没有像 int32 -> int64 这样的自动类型转换,如果你真的想将 []string 作为 []interface{} 发送,你被迫创建一个 []interface{} 副本是有道理的。这样就不会令人惊讶了 - 你明确地写了它,你知道你在做什么。如果函数会修改 []interface{} - 它不会伤害原始的 []string。
TA贡献1921条经验 获得超9个赞
因为[]string
和[]interface{}
有不同的内存布局。当您意识到一个interface{}
变量需要知道它所包含的值的类型时,这一点很明显。
对于[]string
切片,后备数组只需要保存单个字符串。对于[]interface{}
切片,您已经获得了类型信息和字符串值(嗯,指向字符串值的指针,因为字符串大于内存中的单个字)。因此,从一种类型转换为另一种类型将涉及复制数据。
Go 自动执行转换会令人困惑,因为它会使代码推理变得困难。例如,函数调用f(s)
可以修改切片中的字符串,s
如果它被声明为接受一个[]string
参数,但如果它被声明为接受一个参数,则不能[]interface{}
。
- 2 回答
- 0 关注
- 812 浏览
添加回答
举报