在 Go 1.8 中,插件可以使用自定义接口。但是,有一些限制需要考虑。
一种方法是在外部包中定义接口,然后让插件和主应用程序导入它。
例如,在名为filter的包中创建一个接口:
package filter type Filter interface { Name() string Age() int }
在插件中,导入filter包并实现接口:
package main import ( "fmt" "filter" ) type plgFilter struct{} func (plgFilter) Name() string { return "Bob" } func (plgFilter) Age() int { return 23 } func GetFilter() (f filter.Filter, err error) { f = plgFilter{} fmt.Printf("[plugin GetFilter] Returning filter: %T %v\n", f, f) return }
在主应用程序中,导入过滤器包并加载插件:
package main import ( "fmt" "filter" "plugin" ) func main() { p, err := plugin.Open("pg/pg.so") if err != nil { panic(err) } GetFilter, err := p.Lookup("GetFilter") if err != nil { panic(err) } filter, err := GetFilter.(func() (filter.Filter, error))() fmt.Printf("GetFilter result: %T %v %v\n", filter, filter, err) fmt.Println("\tName:", filter.Name()) fmt.Println("\tAge:", filter.Age()) }
这种方法确保插件可以访问接口定义。
另一个选项是让插件返回接口{}类型的值。然后,主应用程序可以定义它期望的接口,并对返回值使用类型断言。
例如,在插件中:
package main import ( "fmt" ) type plgFilter struct{} func (plgFilter) Name() string { return "Bob" } func (plgFilter) Age() int { return 23 } func GetFilterIface() (f interface{}, err error) { f = plgFilter{} fmt.Printf("[plugin GetFilterIface] Returning filter: %T %v\n", f, f) return }
在主应用程序中:
package main import ( "fmt" "plugin" ) func main() { p, err := plugin.Open("pg/pg.so") if err != nil { panic(err) } GetFilterIface, err := p.Lookup("GetFilterIface") if err != nil { panic(err) } filterIface, err := GetFilterIface.(func() (interface{}, error))() fmt.Printf("GetFilterIface result: %T %v %v\n", filterIface, filterIface, err) myfilter := filterIface.(MyFilter) fmt.Println("\tName:", myfilter.Name()) fmt.Println("\tAge:", myfilter.Age()) } type MyFilter interface { Name() string Age() int }
这种方法提供了更大的灵活性,但需要在主程序中进行类型断言application.
请注意,自定义接口仅在插件外部定义的情况下才有效。这是因为 Go 插件是独立的模块,无法访问其他包中定义的类型。
以上是如何通过 Go 1.8 插件使用自定义接口?的详细内容。更多信息请关注PHP中文网其他相关文章!