# [Go Plugin(go 动态库)](https://blog.csdn.net/m0_38132420/article/details/68496881) - [Go Plugin(go 动态库)](#go-plugingo-动态库) - [1. 简介](#1-简介) - [2. 使用](#2-使用) - [3. 注意](#3-注意) ## 1. 简介 在 Go 1.8 出现以前,一直觉得 Go 语言不能像 C/C++一样可以使用动态库的方式动态修改服务。每次升级操作都不得不重新编译整个工程,重新部署服务。这对于很多比较重型的服务来说是一个很致命的弱点。 目前在 Go 1.8 只在 Linux 和 Darwin 系统下支持 Plugin。从 Go 1.8 源码中 plugin 包中 plugin.go 文件开头中有对应的说明。在 Go 1.8 中 plugin 包在操作系统的支持并不十分完善,即使在 Linux 系统下也需要特定 gcc 的编译器及连接器的支持。后续版本应该会有做相应的改进。 ## 2. 使用 创建一个提供方法的文件 print.go ```go package main import ( "fmt" ) func PrintTest(strInput string) { fmt.Println("string in print.so is:", strInput) } ``` 编译 Go 动态库命令: ```bash go build -trimpath -buildmode=plugin -ldflags="-w -s" ``` 其中 -trimpath 用于从二进制文件中删除文件系统路径信息,如果不增加该参数,主程序加载插件时会去匹配 GOPATH,如果二者的 GOPATH 不一致则不进行加载。 指定文件编译 Go 动态库命令: ```bash go build -trimpath -buildmode=plugin -ldflags="-w -s" -o print.so print.go ``` go 动态库方法的使用(main.go): ```go package main import ( "plugin" ) func main() { // 打开动态库 pdll, err := plugin.Open("print.so") if err != nil { //... return } // 获取动态库方法 funcPrint, err := pdll.Lookup("PrintTest") if err != nil { //... return } // 动态库方法调用 funcPrint.(func(string))("hello go plugin") return } ``` 编译: ```bash go build -trimpath -ldflags="-w -s" ``` Go 1.8 中 plugin 包只提供 Open 和 Lookup 方法,对应打开动态库和获取动态库中的方法。注意获取到的动态动态库方法其实是一个 interface{} 类型,需要将其进行转换后才可以使用。 ## 3. 注意 1. 插件实现和主应用程序都必须使用完全相同的 Go 工具链版本构建。 2. 除非指定 -trimpath 参数,否则插件实现和主应用程序都必须使用完全相同的 GOPATH 构建。 3. 插件中的任何依赖项应该与主应用程序中的依赖项版本相同。 4. 你无法将插件编译为静态二进制文件。