本文转载自微信公众号「脑子进煎鱼了」,作者陈煎鱼。转载本文请联系脑子进煎鱼了公众号。
成都创新互联长期为1000多家客户提供的网站建设服务,团队从业经验10年,关注不同地域、不同群体,并针对不同对象提供差异化的产品和服务;打造开放共赢平台,与合作伙伴共同营造健康的互联网生态环境。为介休企业提供专业的成都网站设计、成都网站制作、外贸网站建设,介休网站改版等技术服务。拥有十载丰富建站经验和众多成功案例,为您定制开发。
大家好,我是煎鱼。
在 Go 语言中,有一个好像很好用,但却比较少人提及的功能,那就是 Go Plugin。
目前在 Go 工程中普遍还没广泛的使用起来,覆盖率不高。在 Go issues 上吐槽挺多,甚至感觉有点像凉了半截的样子。
前段时间小咸鱼的同事问了他这功能怎么用,他正想甩出一个链接,但发现...煎鱼竟然没写过,这不,Go 知识板块的文章地图得补全。
今天煎鱼就大家一起学习 Go Plugin,看看为什么会说感觉 “有点像凉了半截” 的样子,打开来看看这个问题在哪。
Go Team 最早在 Go1.7 实验,在 Go1.8 正式引入了 Go Plugin 的机制。于 2016 年发布,一开始仅支持 Linux 实现:
Go Plugin 机制实现了 Go 插件的加载和符号解析,能够支持将我们所编写的 Go 包编译为共享库(.so)。
这样 Go 工程就可以加载所编译好的 Go Plugin(已经变成了共享库文件),在程序中调用共享库中的函数、常量、变量等使用。也称其为 Go 语言中的热插拔的插件系统。
截止 Go1.17 为止,Go Plugin 仅支持在 Linux、FreeBSD 和 MacOS 上运行,还不支持 Windows。
Go 语言是静态语言,正常我们写一个程序,分如下两个角度来看:
那么为什么需要 Go Plugin 呢,原因如下:
其实本质上还是希望程序能够在运行时实现动态的外部加载,根据不同的条件、场景加载不同的插件功能。
Go 官方给出的例子非常简单,只需要在 Go 编译时指定为插件就可以了。
编译的命令例子如下:
- go build -buildmode=plugin
当一个插件初次被打开时,所有尚未成为程序一部分的包的init函数被调用。不过主函数不被运行。需要注意一个插件只会被初始化一次,插件不能被关闭。
其共有如下几个 API:
- type Plugin
- func Open(path string) (*Plugin, error)
- func (p *Plugin) Lookup(symName string) (Symbol, error)
- type Symbol
主要就是细分为插件和符号,符号(Symbol)本身是一个 interface,在调用 Plugin 相关方法后还是需要进一步断言才能使用。
了解基本定义后,我们定义一个插件,一般我们会有个 plugins/ 的目录,作为主程序的附属插件集。
插件的代码如下:
- package main
- import "fmt"
- var V int
- func F() {
- fmt.Printf("脑子进了 %d 次煎鱼 \n", V)
- }
包名必须为 main,在该插件根目录运行:
- go build -buildmode=plugin -o plugin.so main.go
就可以看到在编译的目录下多出了 plugin.so 文件,这就是这个插件经过编译后的动态库 .so 文件。
随后只需在主程序加载这个插件就可以了,如下:
- import (
- "plugin"
- )
- func main() {
- p, err := plugin.Open("plugin.so")
- if err != nil {
- panic(err)
- }
- v, err := p.Lookup("V")
- if err != nil {
- panic(err)
- }
- f, err := p.Lookup("F")
- if err != nil {
- panic(err)
- }
- *v.(*int) = 999
- f.(func())()
- }
输出结果:
- 脑子进了 999 次煎鱼
在程序中,我们先调用了 plugin.Open 方法打开了前面所编译的 plugin.so 动态库。
紧接着调用 plugin.Lookup 方法,定位到了变量 V 和 方法 F,但由于其返回值都是 Symbol(interface),因此我们需要对其进行类型断言,随时才可以调用和使用。
至此完成了一个插件的基本使用。
在前面我们提到了大量 Go Plugin 的优点,也演示了其 Plugin 代码编写起来有多么的简单和方便。
但,为什么 Go Plugin 已经发布了 4 年依然没有被大规模应用,甚至对于不少业务开发来讲是不被需要的呢,或是压根不知道有这东西?
究其原因,我个人认为一个东西的广泛应用要至少符合以下三大点:
比较折腾的人的是,Go Plugin 这三大点都欠一些火候,综合导致了该功能的没有大规模应用。
像是要应用 Go Plugin 有诸如下约束:
在 Go issues 中畅游时,能看到许多小伙伴在以往 4 年踩过的坑和无奈。甚至有一个高赞回答(#19282)表示:插件功能主要是一个技术演示,由于一些不道德的原因,被作为语言的稳定功能发布(The plugin feature is mostly a tech demo that for some unholy reason got released as a stable feature of the language.)。
目前 Go Plugin 并不是 Go Team 的优先事项,在 Windows/Mac 的支持存在问题。GOPATH 有问题,不同 GO 版本也有问题。更是建议如果您想要插件,请走较慢的 grpc 路线,因为它们是有效的插件。
也可以参考为数不多的一些 Go Plugin 用户的方案,例如:tidb,甚至写了个指导文档。
但如果要在生产正式使用,劝你还是需要慎重考虑,又或是再等等...等更完善的那一天?
参考
Go Package plugin
Why is there no windows support for plugins?
plugin: add Windows support
plugin: Add support for closing plugins
如何评价 Go 标准库中新增的 plugin 包?
一文搞懂Go语言的plugin
当前标题:Go插件系统,一个凉了快半截的特性?
文章地址:http://www.shufengxianlan.com/qtweb/news33/74133.html
网站建设、网络推广公司-创新互联,是专注品牌与效果的网站制作,网络营销seo公司;服务项目有等
声明:本网站发布的内容(图片、视频和文字)以用户投稿、用户转载内容为主,如果涉及侵权请尽快告知,我们将会在第一时间删除。文章观点不代表本网站立场,如需处理请联系客服。电话:028-86922220;邮箱:631063699@qq.com。内容未经允许不得转载,或转载时需注明来源: 创新互联