下面我们自己在 Linux 下做一个动态库(.so 文件 - Shared Object),然在用 Go 来使用它。本文所用的操作系统为 Ubuntu18.04, 以 gcc 作为编译器。
1.实现头文件,声明文件中函数。这里创建一个add.h文件。
#ifndef __ADD_H__ #define __ADD_H__ char* Add(char* src, int n); #endif
2.实现add主体函数add.c
#include <string.h> #include <stdio.h> #include <stdlib.h> char* Add(char* src, int n) { char str[20]; sprintf(str, "%d", n); char *result = malloc(strlen(src)+strlen(str)+1); strcpy(result, src); strcat(result, str); return result; }
3.用命令生成动态库,在linux下文件名称是libadd.so
gcc -fPIC -shared -o lib/libadd.so include/add.c
会在当前目录下生成 libadd.so 文件, 在 Linux 下可用 nm -D libadd.so 查看其中的方法
4.编写一个库来测试一下
#include <stdio.h> #include "add.h" int main(int argc, char *argv[]) { char* aa = "giter"; printf("%s\n", Add(aa, 8)); return 0; }
链接动态库生成可执行文件
gcc include/test.c -L lib/ -ladd -o test
- -L .表示搜索要链接的库文件时包含当前目录
- -ladd 表示要链接动态库 libadd.so (备注:默认lib + xxx + .so ,中间的xxx就是库名)
- -o test 生成可执行文件 test
错误:运行出错的情况
# 运行 ./test,出错
./test: error while loading shared libraries: libadd.so: cannot open shared object file: No such file or directory
出现以上的错误。有可能是环境变量没弄好导致的,找不到动态库 libadd.so, Linux 通过 ldconfig 的指示在某些目录中(如 /lib, /user/lib) 搜索动态库。更简单的办法是用 LD_LIBRARY_PATH 环境变量。解决办法,
$ LD_LIBRARY_PATH=lib/ ./test giter8
至此,动态库 libadd.so 准备好了,并且用 test 验证了它是可用的,接下来就在 Go 语言中使用该动态库的函数。
5.golang调用c动态库
demo1 ├── include │ └── add.c │ └── add.h │ └── test.c ├── lib │ └── libadd.so └── main.go
main.go 的代码如下:
package main /* // 头文件的位置,相对于源文件是当前目录,所以是 .,头文件在多个目录时写多个 #cgo CFLAGS: ... #cgo CFLAGS: -I./include // 从哪里加载动态库,位置与文件名,-ladd 加载 libadd.so 文件 #cgo LDFLAGS: -L./lib -ladd -Wl,-rpath,lib #include "add.h" */ import "C" import "fmt" func main() { val := C.Add(C.CString("go"), 2021) fmt.Println("run c: ", C.GoString(val)) }
通过注释代码来告诉 Go 编译器从哪里引入头文件与加载动态库. 本例中 *.h 和 *.go 文件在同一个目录的情况下, #cgo CFLAGS: -I. 可不写。
CFLAGS: -I 和 LDFLAGS: -L 都是相对于源文件 main.go 的位置
./demo1 run c: go2021
成功调用 C 实现的 add 函数
下面列出一些问题
import "C" 要紧挨着 /*...*/ 注释块,如果写成
/* #cgo ... */ import "C"
出现下面的报错信息
# demo1
./main.go:15:10: could not determine kind of name for C.Add
import "C" 要独占一行, 试图同时引入其他的库,如 import ("C"; "fmt") 也会报上面同样的错误
加载不到头文件的错误很明显,#include "add.h" 时会告诉你该文件不存在,如果没有加载到正确的头文件调用 C.Add() 函数时就会报错
# demo1
./main.go:15:10: could not determine kind of name for C.Add
还有一个关键是能否加载到动态库 libadd.so, 参考了网上一些例子,如果把第五行改为
cgo LDFLAGS: -L./lib -ladd
编译不会报错,执行时会出错。
./demo1: error while loading shared libraries: libadd.so: cannot open shared object file: No such file or directory
但如果设置了环境变量 LD_LIBRARY_PATH=/home/vagrant/testgo/lib 也能让它跑起来
LD_LIBRARY_PATH=lib/ ./demo1