一尘不染

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

go

我知道可以查找导出的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 ,我得到一个执行时间错误。在此示例中,“错误的符号类型” __

有没有办法做到这一点,还是在插件和应用程序之间共享数据的唯一方法是使用接口?

谢谢。


阅读 232

收藏
2020-07-02

共1个答案

一尘不染

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是非指针类型的变量,但是我们从导出符号中对指针类型进行了类型声明。这是因为如果您查找 变量
,您将获得指向它的指针,否则您将无法修改变量的值,而只能修改副本。在此答案中对此进行了详细说明:插件符号作为函数返回

如果我们查找插件导出的另一个符号,则不是这种情况:该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插件依赖项如何工作?

2020-07-02