原文地址, songleo

本文主要介绍go语言动态库的编译和使用方法,以linux平台为例,windows平台步骤一样,具体环境如下:

1
2
3
4
5
6
7
8
9
10
11
$ echo $GOPATH
/media/sf_share/git/go_practice
$ echo $GOROOT
/usr/lib/golang/
$ tree $GOPATH/src
/media/sf_share/git/go_practice/src
|-- demo
|   `-- demo.go
`-- main.go

1 directory, 2 files

$GOPATH/src 目录,有 demo 包和使用 demo 包的应用程序 main.gomain.go 代码如下:

1
2
3
4
5
6
7
package main

import "demo"

func main() {
    demo.Demo()
}

demo 包中的 demo.go 代码如下:

1
2
3
4
5
6
7
package demo

import "fmt"

func Demo() {
    fmt.Println("call demo ...")
}

由于 demo.go$GOPATH/src 目录下的一个包, main.go 在 import 该包后,可以直接使用,运行 main.go

1
2
$ go run main.go
call demo ...

现在,需要将 demo.go 编译成动态库 libdemo.so,让 main.go 以动态库方式编译,详细步骤如下:

1 将go语言标准库编译成动态库

1
$ go install -buildmode=shared std

在命令行运行 go install -buildmode=shared std 命令, -buildmode 指定编译模式为共享模式, -linkshared 表示链接动态库,成功编译后会在 $GOROOT 目录下生标准库的动态库文件 libstd.so ,一般位于 $GOROOT/pkg/linux_amd64_dynlink 目录:

1
2
3
$ cd $GOROOT/pkg/linux_amd64_dynlink
$ ls libstd.so
libstd.so

2 将demo.go编译成动态库

1
2
3
4
$ go install  -buildmode=shared -linkshared demo
$ cd $GOPATH/pkg
$ ls linux_amd64_dynlink/
demo.a  demo.shlibname  libdemo.so

成功编译后会在 $GOPATH/pkg 目录生成相应的动态库 libdemo.so

3 以动态库方式编译main.go

1
2
3
4
5
6
7
8
9
10
$ go build -linkshared main.go
$ ll -h
total 25K
drwxrwx---. 1 root vboxsf 4.0K Apr 28 17:30 ./
drwxrwx---. 1 root vboxsf 4.0K Apr 28 17:22 ../
drwxrwx---. 1 root vboxsf    0 Apr 28 08:37 demo/
-rwxrwx---. 1 root vboxsf  16K Apr 28 17:30 main*
-rwxrwx---. 1 root vboxsf   58 Apr 28 08:37 main.go*
$ ./main
call demo ...

从示例中可以看到,以动态库方式编译生成的可执行文件 main 大小才 16K。如果以静态库方式编译,可执行文件main大小为 1.5M ,如下所示:

1
2
3
4
5
6
7
8
9
10
$ go build main.go
$ ll -h
total 1.5M
drwxrwx---. 1 root vboxsf 4.0K Apr 28 17:32 ./
drwxrwx---. 1 root vboxsf 4.0K Apr 28 17:22 ../
drwxrwx---. 1 root vboxsf    0 Apr 28 08:37 demo/
-rwxrwx---. 1 root vboxsf 1.5M Apr 28 17:32 main*
-rwxrwx---. 1 root vboxsf   58 Apr 28 08:37 main.go*
$ ./main
call demo ...

以动态库方式编译时,如果删除动态库 libdemo.so 或者动态库 libstd.so,运行 main 都会由于找不到动态库导致出错,例如删除动态库 libdemo.so

1
2
3
$ rm ../pkg/linux_amd64_dynlink/libdemo.so
$ ./main
./main: error while loading shared libraries: libdemo.so: cannot open shared object file: No such file or directory

以上就是go语言动态库的编译和使用方法,需要注意的是,其他go程序在使用go动态库时,必须提供动态库的源码,否则会编译失败。例如,这里将 demo.go 代码删除,再以动态库方式编译 main.go 时,会编译失败:

1
2
3
4
$ go install  -buildmode=shared -linkshared demo
$ rm demo/demo.go
$ go build -linkshared main.go
main.go:3:8: no buildable Go source files in /media/sf_share/git/go_practice/src/demo

动态库编译方式和静态库不一样,静态库可以不提供源码,直接使用静态库编译,而动态库不行。

References