如何在切片反射中添加元素?

时间:2016-06-21 08:36:20

标签: arrays go insert append

您好我正在学习Go,我正在做一些反思。我陷入了这样的情况:

  1. 我想创建一个slice struct传递给interface{}
  2. 然后我想将新元素附加到此切片
  3. 以下是带有代码示例的playground

    package main
    
    import (
        "fmt"
        "reflect"
    )
    
    type A struct{ Name string }
    
    func main() {
        bbb(A{})
    }
    
    func aaa(v interface{}) {
        sl := reflect.ValueOf(v).Elem()
        typeOfT := sl.Type()
    
        ptr := reflect.New(typeOfT).Interface()
        s := reflect.ValueOf(ptr).Elem()
        sl.Set(reflect.Append(sl, s))
    
        ptr = reflect.New(typeOfT).Interface()
        s = reflect.ValueOf(ptr).Elem()
        sl.Set(reflect.Append(sl, s))
    }
    
    func bbb(v interface{}) {
        myType := reflect.TypeOf(v)
        models := reflect.Zero(reflect.SliceOf(myType)).Interface()
        aaa(&models)
    
        fmt.Println(models)
    }
    

    错误: panic: reflect: call of reflect.Append on interface Value

    有没有办法让它发挥作用? 注意:我想要参考。

    解决方案:

    以下是我的目标:playground

2 个答案:

答案 0 :(得分:1)

当我使用reflect.Zero创建slie时,问题出在reflect.New。这里有完整的工作示例。

https://play.golang.org/p/3mIEFqMxk-

package main

import (
    "encoding/json"
    "fmt"
    "reflect"
)

type A struct {
    Name string `column:"email"`
}

func main() {
    bbb(&A{})
}

func aaa(v interface{}) {
    t := reflect.TypeOf(v)
    if t.Kind() == reflect.Ptr {
        t = t.Elem()
    }

    if t.Kind() == reflect.Slice {
        t = t.Elem()
    } else {
        panic("Input param is not a slice")
    }

    sl := reflect.ValueOf(v)

    if t.Kind() == reflect.Ptr {
        sl = sl.Elem()
    }

    st := sl.Type()
    fmt.Printf("Slice Type %s:\n", st)

    sliceType := st.Elem()
    if sliceType.Kind() == reflect.Ptr {
        sliceType = sliceType.Elem()
    }
    fmt.Printf("Slice Elem Type %v:\n", sliceType)

    for i := 0; i < 5; i++ {
        newitem := reflect.New(sliceType)
        newitem.Elem().FieldByName("Name").SetString(fmt.Sprintf("Grzes %d", i))

        s := newitem.Elem()
        for i := 0; i < s.NumField(); i++ {
            col := s.Type().Field(i).Tag.Get("column")
            fmt.Println(col, s.Field(i).Addr().Interface())
        }

        sl.Set(reflect.Append(sl, newitem))
    }
}

func bbb(v interface{}) {
    t := reflect.TypeOf(v)
    if t.Kind() == reflect.Ptr {
        t = t.Elem()
    }
    if t.Kind() != reflect.Struct {
        panic("Input param is not a struct")
    }

    models := reflect.New(reflect.SliceOf(reflect.TypeOf(v))).Interface()

    aaa(models)

    fmt.Println(models)

    b, err := json.Marshal(models)
    if err != nil {
        panic(err)
    }
    fmt.Println(string(b))
}

答案 1 :(得分:0)

这样做你想要的吗? (playground link

package main

import (
    "fmt"
    "reflect"
)

type A struct{ Name string }

func main() {
    bbb(A{})
}

func aaa(v interface{}) interface{} {
    sl := reflect.Indirect(reflect.ValueOf(v))
    typeOfT := sl.Type().Elem()

    ptr := reflect.New(typeOfT).Interface()
    s := reflect.ValueOf(ptr).Elem()
    return reflect.Append(sl, s)
}

func bbb(v interface{}) {
    myType := reflect.TypeOf(v)
    models := reflect.MakeSlice(reflect.SliceOf(myType), 0, 1).Interface()
    fmt.Println(aaa(models))
}

请注意,我正在返回新切片。我认为你需要另一层间接(指向指针的指针)才能在没有返回值的情况下执行此操作,但这是我第一次在Go中进行反射,所以我可能会发生错误。