附加到作为空接口传递的 golang 切片
Append to golang slice passed as empty interface
如何附加到空接口(已验证为 *[]结构)?
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)
}
}
我尝试过的事情:
这只是恐慌:反映:在 ptr 值上调用 reflect.Append(因为我们传递 &mySlice 而不是 mySlice)
dest = reflect.Append(reflect.ValueOf(dest), reflect.ValueOf(castedModelRow))
这有效,但没有将值设置回 dest...在主函数中,调用解码函数后 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)
}
这是修复问题中显示的第二个 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)
}
您与最初的解决方案非常接近。在调用附加操作之前,您必须取消引用指针。如果您的 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)
如何附加到空接口(已验证为 *[]结构)?
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)
}
}
我尝试过的事情:
这只是恐慌:反映:在 ptr 值上调用 reflect.Append(因为我们传递 &mySlice 而不是 mySlice)
dest = reflect.Append(reflect.ValueOf(dest), reflect.ValueOf(castedModelRow))
这有效,但没有将值设置回 dest...在主函数中,调用解码函数后 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)
}
这是修复问题中显示的第二个 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)
}
您与最初的解决方案非常接近。在调用附加操作之前,您必须取消引用指针。如果您的 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)