main包中定义的标识符不能从其他包中引用,因此从插件导出的标识符不能与您的主应用中的标识符相同。即使您要
Person在插件的文件和主应用程序中都复制该类型,类型断言也会失败,因为它们不是同一类型!
但是可以在单独的程序包中定义类型,并在插件和主应用程序中使用相同的程序包。然后,您可以从在插件中查找的符号中键入断言这种类型。
请参阅以下示例:
在自己的包中定义的单独类型:
package filtertype Filter struct { Name string Age int}插件代码:
package mainimport ( "play/filter")var MyFilter = filter.Filter{ Name: "Bob", Age: 21,}func CreateFilter() filter.Filter { return filter.Filter{ Name: "Bob", Age: 21, }}主要应用:
package mainimport ( "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("%+vn", f)}运行主应用程序,输出为:
&{Name:Bob Age:21}您可能会注意到,插件中的导出标识符
MyFilter是非指针类型的变量,但是我们从导出符号中对指针类型进行了类型声明。这是因为如果您查找 变量
,您将获得指向它的指针,否则您将无法修改变量的值,而只能修改副本。在此答案中对此进行了详细说明:插件符号作为函数返回
如果我们查找插件导出的另一个符号,则不是这种情况:该
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("%+vn", f2)运行此代码,输出将是:
{Name:Bob Age:21}还要注意,如果您更改了
filter插件和主应用程序通常使用的软件包,则还必须重新构建插件。尝试在不重建插件的情况下运行应用程序将导致
plugin.Open()通话过程中出错。有关详细信息,请参阅Go插件依赖项如何工作?



