【问题标题】:why []string can not be converted to []interface{} in golang [duplicate]为什么 []string 不能在 golang 中转换为 []interface{} [重复]
【发布时间】:2014-02-15 00:04:58
【问题描述】:

我觉得很奇怪,为什么[]string不能转换为[]interface{}?

我认为应该可以,因为:

  1. 都是切片
  2. []string的每个元素都是字符串,当然是interface{}

但在下面的例子中,这将是一个编译错误

func f(args ...interface{}){

}
s := []string{"ssd", "rtt"}
f(s...)

为什么语言不能自动完成转换?

【问题讨论】:

标签: types interface go


【解决方案1】:

因为[]string[]interface{} 有不同的内存布局。当您意识到interface{} 变量需要知道它包含的值的类型时,这一点就很明显了。

对于[]string 切片,后备数组只需要保存单个字符串。对于[]interface{} 切片,您已经获得了类型信息和字符串值(嗯,指向字符串值的指针,因为字符串大于内存中的单个单词)。因此,从一种类型转换为另一种类型将涉及复制数据。

Go 自动执行转换会让人感到困惑,因为这会使推理代码变得困难。例如,函数调用f(s) 可以修改切片s 中的字符串,如果它被声明为采用[]string 参数,但如果它被声明为采用[]interface{} 参数,则不能。

【讨论】:

    【解决方案2】:

    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。

    【讨论】:

      猜你喜欢
      • 2017-10-17
      • 2021-05-11
      • 1970-01-01
      • 2015-01-14
      • 2014-12-30
      • 1970-01-01
      • 2012-02-16
      • 1970-01-01
      相关资源
      最近更新 更多