附加到作为空接口传递的 golang 切片

Posted

技术标签:

【中文标题】附加到作为空接口传递的 golang 切片【英文标题】:Append to golang slice passed as empty interface 【发布时间】:2021-12-20 10:13:26 【问题描述】:

如何追加到空接口(已验证为*[]结构)?

func main() 
  var mySlice []myStruct // myStruct can be any struct (dynamic)
  decode(&mySlice, "...")


func decode(dest interface, src string) 
  // assume dest has been verified to be *[]struct
  var modelType reflect.Type = getStructType(dest)
  rows, fields := getRows(src)
  for _, row := range rows 
    // create new struct of type modelType and assign all fields
    model := reflect.New(modelType)
    for field := fields 
      fieldValue := getRowValue(row, field)
      model.Elem().FieldByName(field).Set(fieldValue)
    
    castedModelRow := model.Elem().Interface()
      
    // append model to dest; how to do this?
    // dest = append(dest, castedModelRow)
  

我尝试过的事情:

这简直是恐慌:reflect: 在 ptr 值上调用 reflect.Append(因为我们通过 &mySlice 而不是 mySlice)

dest = reflect.Append(reflect.ValueOf(dest), reflect.ValueOf(castedModelRow))

这可行,但不会将值设置回 dest... 在 main func 中,调用 decode 函数后 len(mySlice) 保持为 0。

func decode(dest interface, src string) 
  ...
  result := reflect.MakeSlice(reflect.SliceOf(modelType), rowCount, rowCount)
  for _, row : range rows 
    ...
    result = reflect.Append(result, reflect.ValueOf(castedModelRow))
  
  dest = reflect.ValueOf(result)

【问题讨论】:

【参考方案1】:

这是修复问题中显示的第二个decode 函数的方法。声明

dest = reflect.ValueOf(result)

修改局部变量dest,而不是调用者的值。使用以下语句修改调用者的切片:

reflect.ValueOf(dest).Elem().Set(result)

问题中的代码在 reflect.MakeSlice 中创建的元素之后附加了解码的元素。生成的切片具有len(rows) 零值,后跟len(rows) 解码值。通过更改修复

result = reflect.Append(result, reflect.ValueOf(castedModelRow))

到:

result.Index(i).Set(model)

这是问题中第二个decode函数的更新版本:

func decode(dest interface, src string) 
    var modelType reflect.Type = getStructType(dest)
    rows, fields := getRows(src)
    result := reflect.MakeSlice(reflect.SliceOf(modelType), len(rows), len(rows))
    for i, row := range rows 
        model := reflect.New(modelType).Elem()
        for _, field := range fields 
            fieldValue := getRowValue(row, field)
            model.FieldByName(field).Set(fieldValue)
        
        result.Index(i).Set(model)
    
    reflect.ValueOf(dest).Elem().Set(result)

Run it on the Playground.

【讨论】:

非常感谢,这就像一个魅力!非常感谢您的解释,并请您指出附加问题。这个写得很好的反思答案当然适合您的用户名句柄:)【参考方案2】:

您与最初的解决方案非常接近。在调用附加操作之前,您必须取消引用指针。如果您的 dest 已经有一些现有元素并且您不想通过创建 newSlice 来丢失它们,则此解决方案会很有帮助。

tempDest := reflect.ValueOf(dest).Elem()
tempDest = reflect.Append(tempDest, reflect.ValueOf(model.Interface()))

与@I Love Reflection 指出的类似,您最终需要将新切片设置回指针。

reflect.ValueOf(dest).Elem().Set(tempDest)

整体解码:

var modelType reflect.Type = getStructType(dest)
rows, fields := getRows(src)
tempDest := reflect.ValueOf(dest).Elem()
for _, row := range rows 
    model := reflect.New(modelType).Elem()
    for _, field := range fields 
        fieldValue := getRowValue(row, field)
        model.FieldByName(field).Set(fieldValue)
    
    tempDest = reflect.Append(tempDest, reflect.ValueOf(model.Interface()))

reflect.ValueOf(dest).Elem().Set(tempDest)

【讨论】:

在这种情况下,我不想附加到 dest 中的现有值,而是要覆盖它,是的: reflect.ValueOf(dest).Elem().Set(result) 可以解决问题。非常感谢!在问这个问题之前,我最初尝试过 reflect.ValueOf(dest).Set(result) ......并感到恐慌:reflect.Value.Set using unaddressable value。这似乎是因为 func (v Value) Set(x Value) 分配了 v.ptr = x.ptr 所以这就是为什么我们需要做 .Elem() 所以 Set 函数接收值而不是 ptr?

以上是关于附加到作为空接口传递的 golang 切片的主要内容,如果未能解决你的问题,请参考以下文章

Golang basic_leaming接口

GoLang接口---中

为啥可以将切片分配给空接口但不能将其强制转换为相同的空接口

使用接口在非空数组(切片)中查找另一个“特殊”数组

golang中Any类型使用及空接口中类型查询

Golang "..."用法