文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

将在 Go 语言切片中应用反射和动态类型

2024-04-04 23:25

关注

一分耕耘,一分收获!既然打开了这篇文章《将在 Go 语言切片中应用反射和动态类型》,就坚持看下去吧!文中内容包含等等知识点...希望你能在阅读本文后,能真真实实学到知识或者帮你解决心中的疑惑,也欢迎大佬或者新人朋友们多留言评论,多给建议!谢谢!

问题内容

我的问题是 https://stackoverflow.com/a/54909356 的后续问题(演示:https://play.golang.org/p/vxoqtvseleo)。

在该答案中,有一个给定的结构(foo)。我发现这个概念很有趣,并且想知道对于动态类型也可以做同样的事情。

鉴于以下动态结构(完美运行),您如何创建切片并附加到它?

在这里查看我的尝试:

在此代码片段中,我创建了动态类型。这工作得很好。

typestring := reflect.typeof("")
        typeinteger := reflect.typeof(0)
        fields := make(map[string]*reflect.structfield)
        fields["a"] = &reflect.structfield{
                name: "a",
                type: typestring,
                tag:  reflect.structtag(`json:"a"`),
        }
        fields["b"] = &reflect.structfield{
                name: "b",
                type: typeinteger,
                tag:  reflect.structtag(`json:"b"`),
        }
        var structfields []reflect.structfield
        for _, v := range fields {
                structfields = append(structfields, *v)
        }

        structtype := reflect.structof(structfields)

使用该结构,我们发现它工作正常:

v := reflect.new(structtype)
        v.elem().fieldbyname("a").setstring("qwer")
        v.elem().fieldbyname("b").setint(123)

        v2 := reflect.new(structtype)
        v2.elem().fieldbyname("a").setstring("asdf")
        v2.elem().fieldbyname("b").setint(456)

        //+ these look good
        fmt.printf("%v\n", v)
        fmt.printf("%v\n", v2)

但是,进入动态切片时,我们收到错误:

reflection := reflect.MakeSlice(reflect.SliceOf(reflect.TypeOf(structType).Elem()), 0, 0)

        reflectionValue := reflect.New(reflection.Type())
        reflectionValue.Elem().Set(reflection)

        slicePtr := reflect.ValueOf(reflectionValue.Interface())

        sliceValuePtr := slicePtr.Elem()

        value := v.Elem()

        // ERROR HERE
        toSet := reflect.Append(sliceValuePtr, value)

需要调整哪些内容才能使所有内容对齐并 append

请参阅演示中的以下内容:

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


解决方案


structtypereflect.type。表达式 reflect.typeof(structtype) 计算 reflect.type 的值为 reflect.type。直接使用 structtype 而不是调用 reflect.typeof(structtype)

typestring := reflect.typeof("")
typeinteger := reflect.typeof(0)
fields := make(map[string]*reflect.structfield)
fields["a"] = &reflect.structfield{
    name: "a",
    type: typestring,
    tag:  reflect.structtag(`json:"a"`),
}
fields["b"] = &reflect.structfield{
    name: "b",
    type: typeinteger,
    tag:  reflect.structtag(`json:"b"`),
}
var structfields []reflect.structfield
for _, v := range fields {
    structfields = append(structfields, *v)
}

structtype := reflect.structof(structfields)

v := reflect.new(structtype)
v.elem().fieldbyname("a").setstring("qwer")
v.elem().fieldbyname("b").setint(123)

v2 := reflect.new(structtype)
v2.elem().fieldbyname("a").setstring("asdf")
v2.elem().fieldbyname("b").setint(456)

//+ these look good
fmt.printf("%v\n", v)
fmt.printf("%v\n", v2)

reflection := reflect.makeslice(reflect.sliceof(structtype), 0, 0)

fmt.printf("typeof structtype: %v\n", structtype)
fmt.printf("typeof structtype | elem: %v\n", reflect.typeof(structtype).elem())

reflectionvalue := reflect.new(reflection.type())
reflectionvalue.elem().set(reflection)

sliceptr := reflect.valueof(reflectionvalue.interface())

slicevalueptr := sliceptr.elem()

value := v.elem()

fmt.printf("value: %t = (%v)\n", value, value)
fmt.printf("slice: %t = (%v)\n", slicevalueptr, slicevalueptr)

toset := reflect.append(slicevalueptr, value)
fmt.printf("toset: %t = (%v)\n", toset, toset)

slicevalueptr.set(toset)

result := slicevalueptr.interface()

fmt.printf("result: %t = (%v)\n", result, result)

Run it on the Go playground。

这是代码的简化版本:

stringType := reflect.TypeOf("")
intType := reflect.TypeOf(0)
t := reflect.StructOf([]reflect.StructField{
    {
        Name: "A",
        Type: stringType,
        Tag:  reflect.StructTag(`json:"a"`),
    },
    {
        Name: "B",
        Type: intType,
        Tag:  reflect.StructTag(`json:"b"`),
    },
})

v1 := reflect.New(t).Elem()
v1.FieldByName("A").SetString("qwer")
v1.FieldByName("B").SetInt(123)

v2 := reflect.New(t).Elem()
v2.FieldByName("A").SetString("asdf")
v2.FieldByName("B").SetInt(456)

slice := reflect.MakeSlice(reflect.SliceOf(t), 0, 0)
slice = reflect.Append(slice, v1, v2)
fmt.Printf("result: %s = (%v)\n", slice.Type(), slice)

Run it on the playground。

以上就是《将在 Go 语言切片中应用反射和动态类型》的详细内容,更多关于的资料请关注编程网公众号!

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     801人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     348人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     311人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     432人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     220人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯