dongqu5650 2018-12-23 15:49
浏览 71
已采纳

golang插件可以用于工厂功能吗?

I have the following code in a golang plugin module:

plug.go

package main

import "fmt"

var (
    Thing        = New("first thing")
    ThingFactory = thingFactory{}
)

type thing struct {
    i int
    s string
}

func New(s string) thing {
    return thing{s: s}
}

func (t *thing) Say() string {
    t.i++
    return fmt.Sprintf("%s - %d", t.s, t.i)
}

type thingFactory struct{}

func (t thingFactory) Make(s string) thing {
    return New(s)
}

it is compiled as a .so object and used in another program:

main.go

package main

import (
    "fmt"
    "plugin"
)

func main() {
    p, err := plugin.Open("../plug/plug.so")
    if err != nil {
        panic(err)
    }
    symbol, err := p.Lookup("Thing")
    if err != nil {
        panic(err)
    }
    thing := symbol.(Sayer)
    fmt.Println(thing.Say())

    symbol, err = p.Lookup("ThingFactory") // <-problems start here
    if err != nil {
        panic(err)
    }
    factory := symbol.(GetSayer)

    madeThing := factory.Make("how about me?")
    fmt.Println(madeThing.Say())
    fmt.Println(madeThing.Say())
}

type Sayer interface {
    Say() string
}

type GetSayer interface {
    Make(string) Sayer
}

I'm able to lookup the Thing, and call Say() on it, but the second interface conversion panics:

first thing - 1 panic: interface conversion: *main.thingFactory is not main.GetSayer: missing method Make

even though the runtime recognizes the first symbol as a Sayer it doesn't recognize that thingFactory obviously has a Make() method, which should return something that is also a Sayer.

Am I missing something obvious here?

  • 写回答

2条回答 默认 最新

  • dougaodi8895 2018-12-23 19:53
    关注

    The first problem is that in your plugin thingFactory (more precicely *thingfactory) does not have a method described in your main app's GetSayer interface:

    Make(string) Sayer
    

    You have:

    Make(string) thing
    

    So (first) you have to change thingFactory.Make() to this:

    type Sayer interface {
        Say() string
    }
    
    func (t thingFactory) Make(s string) Sayer {
        th := New(s)
        return &th
    }
    

    After this it still won't work. And the reason for this is because the plugin's Sayer type is not identical to your main app's Sayer type. But they must be the same in order to implement your main app's GetSayer interface.

    One solution is to "outsource" the Sayer interface to its own package, and use this common, shared package both in the plugin and in the main app.

    Let's create a new package, call it subplay:

    package subplay
    
    type Sayer interface {
        Say() string
    }
    

    Import this package and use it in the plugin:

    package main
    
    import (
        "fmt"
        "path/to/subplay"
    )
    
    var (
        Thing        = New("first thing")
        ThingFactory = thingFactory{}
    )
    
    type thing struct {
        i int
        s string
    }
    
    func New(s string) thing {
        return thing{s: s}
    }
    
    func (t *thing) Say() string {
        t.i++
        return fmt.Sprintf("%s - %d", t.s, t.i)
    }
    
    type thingFactory struct{}
    
    func (t thingFactory) Make(s string) subplay.Sayer {
        th := New(s)
        return &th
    }
    

    And also import and use it in the main app:

    package main
    
    import (
        "fmt"
        "path/to/subplay"
        "plugin"
    )
    
    func main() {
        p, err := plugin.Open("../plug/plug.so")
        if err != nil {
            panic(err)
        }
        symbol, err := p.Lookup("Thing")
        if err != nil {
            panic(err)
        }
        thing := symbol.(subplay.Sayer)
        fmt.Println(thing.Say())
    
        symbol, err = p.Lookup("ThingFactory")
        if err != nil {
            panic(err)
        }
        factory := symbol.(GetSayer)
    
        madeThing := factory.Make("how about me?")
        fmt.Println(madeThing.Say())
        fmt.Println(madeThing.Say())
    }
    
    type GetSayer interface {
        Make(string) subplay.Sayer
    }
    

    Now it will work, and output will be:

    first thing - 1
    how about me? - 1
    how about me? - 2
    

    See related questions:

    go 1.8 plugin use custom interface

    How do Go plugin dependencies work?

    本回答被题主选为最佳回答 , 对您是否有帮助呢?
    评论
查看更多回答(1条)

报告相同问题?

悬赏问题

  • ¥20 有偿 写代码 要用特定的软件anaconda 里的jvpyter 用python3写
  • ¥20 cad图纸,chx-3六轴码垛机器人
  • ¥15 移动摄像头专网需要解vlan
  • ¥20 access多表提取相同字段数据并合并
  • ¥20 基于MSP430f5529的MPU6050驱动,求出欧拉角
  • ¥20 Java-Oj-桌布的计算
  • ¥15 powerbuilder中的datawindow数据整合到新的DataWindow
  • ¥20 有人知道这种图怎么画吗?
  • ¥15 pyqt6如何引用qrc文件加载里面的的资源
  • ¥15 安卓JNI项目使用lua上的问题