GNU/Linux GCC 指定动态库路径 详细讲述了Linux下动态库路径的作用。
在C/C++程序里经常会调用到外部.so文件中的库函数,运行程序时需要export LD_LIBRARY_PATH,不过使用它存在一些弊端,可能会影响到其它程序的运行。在经历的大项目中就遇到过,两个模块同时使用一外部动态库,而且版本还有差异,导致其中一模块出错,两模块是不同时期不同人员分别开发,修正起来费时费力。
对于上述问题,一个比较好的方法是在程序编译的时候加上参数-Wl,-rpath,指定编译好的程序在运行时动态库的目录。这种方法会将动态库路径写入到elf文件中去。 -Wl 的意思是将逗号后面的内容传递给 linker链接器。
-Wa,<options> Pass comma-separated <options> on to the assembler
-Wp,<options> Pass comma-separated <options> on to the preprocessor
-Wl,<options> Pass comma-separated <options> on to the linker
例子:
1.pos.c
#include <stdio.h>
void pos()
{
printf("Hello world, I'm a library!\n");
}
2.main.c
void pos();
int main()
{
pos();
return 0;
}
我们通过以下命令用源程序 pos.c 来创建动态库 libpos.so 。
gcc -fPIC -c pos.c
gcc -shared -fPCI -o libpos.so pos.o
生成程序命令
gcc -o pos main.c -L. -lpos
运行 ./pos 时,发现找不到 libpos.so,怎么会呢?不是在当前目录下吗?是的,linux并不从当前目录加载.so
因此我们需要在编译目标代码时指定可执行文件的动态库搜索路径,所以需要用 gcc 命令重新编译源程序 main.c( 见程序 2) 来生成可执行文件 pos 。
gcc -o pos main.c -L. -lpos -Wl,-rpath=.
再运行程序 pos 试试。
my@systemX:~/go/lib$ ./pos
Hello world, I'm a library!
程序 pos 运行成功,输出的结果正是 pos.c 中的函数 pos 的运行结果。因此程序 pos 搜索到的动态库是当前目录下的 libpos.so 。
参考
https://stackoverflow.com/questions/6562403/i-dont-understand-wl-rpath-wl