登录
首页 >  Golang >  Go问答

通过反射在运行时按名称动态实例化结构

来源:stackoverflow

时间:2024-04-24 08:54:33 439浏览 收藏

最近发现不少小伙伴都对Golang很感兴趣,所以今天继续给大家介绍Golang相关的知识,本文《通过反射在运行时按名称动态实例化结构》主要内容涉及到等等知识点,希望能帮到你!当然如果阅读本文时存在不同想法,可以在评论中表达,但是请勿使用过激的措辞~

问题内容

go 中是否有与 java 等语言提供的动态类实例化功能等效的功能(注意:为简洁起见,此处省略了必需的异常处理逻辑):

class cls = class.forname("org.company.domain.user");
constructor userconstructor = cls.getconstructor();
user user1 = userconstructor.newinstance();

上面的简短 java 代码片段本质上是通过提供的完全限定类路径字符串获取对类的引用,然后使用类引用来获取对零参数构造函数(如果存在)的引用,并且最后,构造函数用于获取对该类实例的引用。

我还没有在 go 中找到可以实现类似结果的类似机制的示例。更具体地说,go 中的 reflect 包似乎要求调用者已经引用了他们希望实例化的结构类型。这方面的标准习惯用法似乎如下:

reflect.New(reflect.TypeOf(domain.User))

注意:提供给reflect.typeof函数的参数必须是类型而不是字符串。可以通过反射包在 go 中实例化结构体,仅使用其完全限定名称吗?


正确答案


kubernetes 在 runtime.Scheme 结构中处理这个确切的过程。这个想法是,您使用名称或其他标识符注册类型,然后您可以根据标识符随意请求这些类型的新实例。一般来说,该标识符是在序列化过程中派生的,而不是硬编码到源中。

问题正如您所说,您首先需要创建一个新实例。虽然这种模式并不常见,但我在职业生涯中遇到过两个案例,这是合乎逻辑的解决方案。以下是 k8s runtime.scheme 为实现此目的所做的非常精简版本的示例,它可能适用于您正在尝试执行的操作和 here it is in action

package main

import (
    "fmt"
    "reflect"
)

type Scheme struct {
    types map[string]reflect.Type
}

func (s *Scheme) RegisterType(name string, t interface{}) {
    a := reflect.TypeOf(t)
    s.types[name] = a
}

func (s *Scheme) New(name string) (interface{}, error) {
    t, ok := s.types[name]
    if !ok {
        return nil, fmt.Errorf("unrecognized type name: %s", name)
    }
    return reflect.New(t).Interface(), nil
}

func NewScheme() *Scheme {
    return &Scheme{types: map[string]reflect.Type{}}
}

type MyType struct {
    Foo string
}

func main() {
    scheme := NewScheme()
    scheme.RegisterType("my.type", MyType{})
    myType, _ := scheme.New("my.type")
    myType.(*MyType).Foo = "bar"
    fmt.Println("%+v", myType)
}

今天关于《通过反射在运行时按名称动态实例化结构》的内容介绍就到此结束,如果有什么疑问或者建议,可以在golang学习网公众号下多多回复交流;文中若有不正之处,也希望回复留言以告知!

声明:本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>