是否可以在go插件和应用程序之间共享自定义数据类型?

时间:2018-05-05 23:21:54

标签: go plugins

我知道可以查找导出的go-plugin符号并将其声明为接口。但是,我想知道是否有一种方法可以将它们键入到结构中,例如。有办法吗?

例如:

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 ,我得到执行时错误。在此示例中,"错误的符号类型"

有没有办法实现这个或者在插件和应用程序之间共享数据的唯一方法是使用接口?

感谢。

1 个答案:

答案 0 :(得分:2)

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中的导出标识符是非指针类型的变量,但我们从导出的符号中键入了一个指针类型。这是因为如果查找变量,您将获得指向它的指针,否则您无法修改变量的值,您只能修改副本。这个答案详细说明:Plugin symbol as function return

如果我们查找插件导出的其他符号,则不是这种情况: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}

请参阅相关问题:go 1.8 plugin use custom interface

另请注意,如果您更改插件和主应用程序通常使用的filter包,则还必须重建插件。尝试在不重建插件的情况下运行应用程序将导致plugin.Open()调用期间出错。有关详细信息,请参阅How do Go plugin dependencies work?