我知道可以查找导出的go-plugin符号,然后将它们断言到 interface中。但是,我想知道是否有一种方法可以将它们断言为一个结构。有办法吗?
例如:
plugin.go
package main type Person struct { Name string } var ( P = Person{ Name: "Emma", } )
app.go
package main import ( "fmt" "plugin" "os" ) func main() { plug, err := plugin.Open("./plugin.so") if err != nil { fmt.Println(err) os.Exit(1) } sym, err := plug.Lookup("P") if err != nil { fmt.Println(err) os.Exit(1) } var p Person p, ok := sym.(Person) if !ok { fmt.Println("Wrong symbol type") os.Exit(1) } fmt.Println(p.Name) }
调用 plug.Lookup 时会找到符号P(即 Person) 。但是,我不能将P输入为 Person ,我得到一个执行时间错误。在此示例中,“错误的符号类型” 。 __
有没有办法做到这一点,还是在插件和应用程序之间共享数据的唯一方法是使用接口?
谢谢。
main包中定义的标识符不能从其他包中引用,因此从插件导出的标识符不能与您的主应用中的标识符相同。即使您要Person在插件的文件和主应用程序中都复制该类型,类型断言也会失败,因为它们不是同一类型!
main
Person
但是可以在单独的程序包中定义类型,并在插件和主应用程序中使用相同的程序包。然后,您可以从在插件中查找的符号中键入断言这种类型。
请参阅以下示例:
在自己的包中定义的单独类型:
package filter type Filter struct { Name string Age int }
插件代码:
package main import ( "play/filter" ) var MyFilter = filter.Filter{ Name: "Bob", Age: 21, } func CreateFilter() filter.Filter { return filter.Filter{ Name: "Bob", Age: 21, } }
主要应用:
package main import ( "fmt" "log" "os" "play/filter" "plugin" ) func main() { p, err := plugin.Open("plugin.so") if err != nil { log.Fatal(err) } mf, err := p.Lookup("MyFilter") if err != nil { log.Fatal(err) } f, ok := mf.(*filter.Filter) if !ok { log.Fatal("Wrong symbol type") } fmt.Printf("%+v\n", f) }
运行主应用程序,输出为:
&{Name:Bob Age:21}
您可能会注意到,插件中的导出标识符MyFilter是非指针类型的变量,但是我们从导出符号中对指针类型进行了类型声明。这是因为如果您查找 变量 ,您将获得指向它的指针,否则您将无法修改变量的值,而只能修改副本。在此答案中对此进行了详细说明:插件符号作为函数返回
MyFilter
如果我们查找插件导出的另一个符号,则不是这种情况:该CreateFilter()函数返回非指针类型的值filter.Filter:
CreateFilter()
filter.Filter
cf, err := p.Lookup("CreateFilter") if err != nil { log.Fatal(err) } createFilter, ok := cf.(func() filter.Filter) if !ok { log.Fatal("Wrong function type") } f2 := createFilter() fmt.Printf("%+v\n", f2)
运行此代码,输出将是:
{Name:Bob Age:21}
还要注意,如果您更改了filter插件和主应用程序通常使用的软件包,则还必须重新构建插件。尝试在不重建插件的情况下运行应用程序将导致plugin.Open()通话过程中出错。有关详细信息,请参阅Go插件依赖项如何工作?
filter
plugin.Open()