在线时间:8:00-16:00
迪恩网络APP
随时随地掌握行业动态
扫描二维码
关注迪恩网络微信公众号
转自:http://studygolang.com/articles/1441 最近需要做一些在go中使用动态C++库的工作,经常碰到找不到动态库路径这种情况,所以就花点时间,专门做一下实验来了解Go。
一、示例代码目录结构(假设代码根目录为/home/gdc/cgotest): ----|bin: #ifndef __API_H__ #define __API_H__ void hello(); #endif hello.go文件内容: package hello /* #cgo CFLAGS: -I./include #cgo LDFLAGS: -L../lib -lapi #include "api.h" */ import "C" func Hello(){ C.hello(); } main.go文件内容: package main import "oidb/hello" func main(){ hello.Hello(); }
A. 将该代码目录添加到GOPATH环境变量中。——export GOPATH=$GOPATH:/home/gdc/cgotest B. 环境变量LD_LIBRARY_PATH值为空
A. 按照如上的目录组织及文件内容,可以在任意地方执行go build oidb/hello或者 go install oidb/hello。 执行go build oidb/hello这个命令,可能会没啥反应,但是可以告诉你编译通过了。 执行go install oidb/hello这个命令,会在代码根目录下的pkg/linux_amd64目录(如果不存在会自动创建)创建oidb/hello.a。 OK,说明该包已经可以给别人使用了。但是别人能否正常使用呢?接下来做一些main.go相关实验。
B. 很遗憾,此时我在很多位置尝试执行go build main 或者 go install main,都会提示我找不到libapi.so动态库。 但有趣的是,当我在lib或hello目录下执行go build main 或者 go install main时,都会成功。其中执行go install main会在代码根目录的bin子目录下生成 一个main可执行文件,go build main会在当前目录下生成一个main可执行程序。然后当我尝试执行这个位于bin目录下的main可执行文件后,会提示找 不到libapi.so这个动态库。 然后我又尝试将lib目录复制到bin目录下和代码根目录下,然后在bin子目录下执行main可执行文件,还是提示找不到libapi.so动态库。 最后,我拿出终极大招,执行"export LD_LIBRARY_PATH=/home/gdc/cgotest/lib"(此时我将lib目录移动到代码根目录下了),然后就可以正常 运行main可执行程序了。 然后我又将该lib目录移动到bin目录和src目录下,然后分别用export命令将这个lib目录的新位置添加到LD_LIBRARY_PATH环境变量中,然后执行main,都可以正常执行。
小结: main包的编译理解: 当编译或安装main包时,go会以此时所处的位置为当前目录。然后如果其引用的某个包中使用相对目录依赖某个动态库C/C++库。那么会从当前目录出发,根据那个相对位置去找动态库。所以上面在编译或安装main包时,唯独在lib或hello目录下成功通过编译了,这就是因为go以从当前目录出发,到其父目录的lib子目录下寻找libapi.so动态库,然后成功找到,从而通过编译。其实不一定非要在lib或者hello目录下编译main包。只要确保编译main包位置的父目录下有一个子目录lib,同时该lib目录下有libapi.so这个文件,即可通过编译。比如,我将lib目录在代码根目录下,然后我转到bin目录下,然后就可以成功编译或安装main包了。 当然了,如果你在使用动态库时使用"-L"选项设置的是绝对目录而不是相对目录,那么编译或安装main包,就没有这么多限制了。你可以在任意位置编译或安装。
D2. 第二种方法:在go中设置环境变量的值。 经过自己实验,然后问魏老大,感觉这是不可能的。因为当程序启动时,系统就会自动加载该程序所依赖的那些库,而此时你在程序中设置环境变量的
代码还没运行呢。当然还是找不到动态库。 一个解决办法:自己手动加载动态库。 参考了http://blog.csdn.net/joker0910/article/details/6103793这篇文章的手动加载库,可以正常使用。
F. 补充(2014.07.21)。 package hello /* #cgo LDFLAGS: -ldl #include #include #include "api.h" void hello_c(char* lib_path){ char* func_name="hello"; void* libc; void (*hello_call)(); if(libc = dlopen(lib_path,RTLD_LAZY)) { hello_call = dlsym(libc, func_name); (*hello_call)(); dlclose(libc); } } */ import "C" import "unsafe" var EXTENSION_DIR string = "/home/guess/.davengu_workdir/go_learning/cgo/use_shared_library/src/oidb/lib/" var OIDB_API string = "libapi.so" //#cgo LDFLAGS: -L/home/guess/.davengu_workdir/go_learning/cgo/use_shared_library/src/oidb/lib -lapi //#cgo LDFLAGS: -L../lib -lapi func Hello() { libPathC := C.CString(EXTENSION_DIR+OIDB_API); defer C.free(unsafe.Pointer(libPathC)); C.hello_c(libPathC); }
|
请发表评论