一尘不染

反射。将结构切片值设置为结构,而无需类型声明(因为它是未知的)

go

我正在创建一个帮助程序包,以从队列中弹出有效负载。至关重要的是,此帮助程序必须与导入它的应用程序所使用的结构无关。

此(无操作,仅作为示例)函数将从队列中提供单个有效载荷,其类型like interface{}如下:

func One(like interface{}) interface{} {
    typ := reflect.TypeOf(like)
    one := reflect.New(typ)

    return one.Interface()
}

此功能提供许多有效负载:

func Many(num int, like interface{}) interface{} {
    typ := reflect.TypeOf(like)
    many := reflect.MakeSlice(reflect.SliceOf(typ), num, num)

    for i := 0; i < num; i++ {
        one := One(typ)
        many.Index(i).Set(one)
    }

    return many.Interface()
}

用法示例是:

type Payload struct {
    Id int
    Text string
}

func main() {
    Many(4, Payload{})
}

但是,以上结果导致:

panic: reflect.Set: value of type **reflect.rtype is not assignable to type main.Payload

https://play.golang.org/p/ud23ZlD3Bx


阅读 256

收藏
2020-07-02

共1个答案

一尘不染

你调用reflect.TypeOf一个reflect.Value,这就是**reflect.rtype来自何方。

直接One使用like值调用函数,然后将结果分配给切片。

func One(like interface{}) interface{} {
    typ := reflect.TypeOf(like)
    one := reflect.New(typ)

    return one.Interface()
}

func Many(num int, like interface{}) interface{} {
    typ := reflect.TypeOf(like)
    many := reflect.MakeSlice(reflect.SliceOf(typ), num, num)

    for i := 0; i < num; i++ {
        one := One(like)
        many.Index(i).Set(reflect.ValueOf(one).Elem())
    }

    return many.Interface()
}

https://play.golang.org/p/fHF_zrymcI

2020-07-02