亚洲在线久爱草,狠狠天天香蕉网,天天搞日日干久草,伊人亚洲日本欧美

為了賬號安全,請及時綁定郵箱和手機立即綁定
已解決430363個問題,去搜搜看,總會有你想問的

附加到作為空接口傳遞的 golang 切片

附加到作為空接口傳遞的 golang 切片

Go
ABOUTYOU 2022-10-10 19:23:51
如何附加到空接口(已驗證為 *[] 結構)?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... 在 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)}
查看完整描述

2 回答

?
素胚勾勒不出你

TA貢獻1827條經驗 獲得超9個贊

這是修復問題中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)

}



查看完整回答
反對 回復 2022-10-10
?
青春有我

TA貢獻1784條經驗 獲得超8個贊

您與原始解決方案非常接近。在調用附加操作之前,您必須取消引用指針。如果您的 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)


查看完整回答
反對 回復 2022-10-10
  • 2 回答
  • 0 關注
  • 141 瀏覽
慕課專欄
更多

添加回答

舉報

0/150
提交
取消
微信客服

購課補貼
聯系客服咨詢優惠詳情

幫助反饋 APP下載

慕課網APP
您的移動學習伙伴

公眾號

掃描二維碼
關注慕課網微信公眾號