如何通过字符串结构名称的字符串名称调用方法?

时间:2017-03-20 14:51:05

标签: go methods dynamic struct

我在struct定义下有一个带有2个方法的结构,我想在其他地方调用,使用struct的名称和方法的名称作为param。

结构代码如下:

type ArticleForm struct {
Name     string     `required:"true" pattern:"^[A-Za-z0-9\u4e00-\u9fa5]{1,1024}$" valid:"Required;MaxSize(1024)"`
Category []Category `class:"multis" required:"true" valid:"Required" optionqs:"GetCategoryOption"`
Content  string     `class:"wysiwg_area" required:"true" valid:"Required"`
Tags     []Tags     `class:"multis_create" optionqs:"GetTagOptions"`
}

方法定义如下:

func (this *ArticleForm) GetTagOptions() []Tags {
return GetTagsOptions(nil)
}

以下是我想要的:

func main() {
s := "models.ArticleForm"
t := "GetTagOptions"
//following is the question, how can i exec following?
funcall(s,t)
}

如何完成funcall(s,t)

1 个答案:

答案 0 :(得分:4)

调用由其名称给出的某种类型的方法很简单(使用reflection)。见这个例子:

type My struct{}

func (m My) MyFunc() {
    fmt.Println("MyFunc called")
}

func main() {
    m := My{}
    meth := reflect.ValueOf(m).MethodByName("MyFunc")
    meth.Call(nil)
}

输出(在Go Playground上尝试):

MyFunc called

"实例"给定其字符串名称的类型是不可能的,因为如果您的代码没有显式引用该类型,它甚至可能不包含在可执行二进制文件中。有关详细信息,请参阅Call all functions with special prefix or suffix in Golang;和Splitting client/server code

一种可行的解决方法是使用某种" type-registry"您在使用它之前填充的(在您想要通过其名称创建值之前)。类型注册表(可以是地图)可以保存从类型名称映射的reflect.Type值或工厂函数。

遵循上述My类型声明,存储reflect.Type值的类型注册表可能如下所示(在Go Playground上尝试):

registry := map[string]reflect.Type{
    "My": reflect.TypeOf((*My)(nil)).Elem(),
}

v := reflect.New(registry["My"]).Elem()
v.MethodByName("MyFunc").Call(nil)

存储工厂功能的注册表可能如下所示(在Go Playground上尝试):

registry := map[string]func() interface{}{
    "My": func() interface{} { return My{} },
}

v := registry["My"]()
meth := reflect.ValueOf(v).MethodByName("MyFunc")
meth.Call(nil)