假设有两个相同的共享库,一个在标准的共享库搜索目录(/lib/i386-linux-gnu), 一个在非标准目录(/home/charles/tmp):
p$ ls /home/charles/tmp/libshared.so /lib/i386-linux-gnu/libshared.so -l -rwxr-xr-x 1 root root 7706 Jan 24 14:56 /home/charles/tmp/libshared.so -rwxrwxr-x 1 charles charles 7706 Jan 24 14:45 /lib/i386-linux-gnu/libshared.so
在/home/charles/tmp下有个测试程序main.c, 调用共享库里的函数。
用如下的命令编译:
gcc main.c -L /home/charles/tmp -g -o main -lshared用ldd看一下link的共享库:
$ ldd main linux-gate.so.1 => (0xb7776000) libshared.so => /lib/i386-linux-gnu/libshared.so (0xb774f000) libc.so.6 => /lib/i386-linux-gnu/libc.so.6 (0xb75a0000) /lib/ld-linux.so.2 (0xb7777000)可以看出,虽然我们指定了 要使用 /home/charles/tmp下的库,但实际上用的还是标准搜索路径下的库。
换句话说,GNU ld 优选搜索标准路径下的文件,然后才是 -L指定的
现在执行
export LD_LIBRARY_PATH=/home/charles/tmp
$ gcc main.c -L /home/charles/tmp -g -o main -lshared $ ldd main linux-gate.so.1 => (0xb7798000) libshared.so => /home/charles/tmp/libshared.so (0xb7792000) libc.so.6 => /lib/i386-linux-gnu/libc.so.6 (0xb75c2000) /lib/ld-linux.so.2 (0xb7799000)可见, LD_LIBRARY_PATH指定的路径优选于系统标准目录。
最后尝试一下如下的方法:
gcc main.c -wl,-rpath,/home/charles/tmp -g -o main -lshared
或者:
gcc main.c -Wl,-rpath /home/charles/tmp -g -o main -lshared
$ ldd main linux-gate.so.1 => (0xb77e0000) libshared.so => /home/charles/tmp/libshared.so (0xb77da000) libc.so.6 => /lib/i386-linux-gnu/libc.so.6 (0xb760a000) /lib/ld-linux.so.2 (0xb77e1000)
最后,总结一下,共享库的搜索顺序(按照优先次序):
1) LD_LIBRARY_PATH指定的目录
2) rpath指定的路径。
rpath指定的路径同时被写进了二进制文件里面。
p$ readelf -d main | grep rpath 0x0000000f (RPATH) Library rpath: [/home/charles/tmp]
3)系统标准路径
这包括 /lib, /usr/lib, /usr/local/lib以及在/etc/ld.conf 里面指定的路径。