准备动态库
在此之前如果你还没有对动态库有一个基本的了解的话,建议你阅读《浅谈静态库和动态库》或其他相关资料。为了说明后面的问题,这里我们先创建一个简单的动态库,你也可以参考《手把手教你制作动态库》:
- // test.c
- //来源:公众号【编程珠玑】
- #include <stdio.h>
- #include "test.h"
- #include "test1.h"
- void test()
- {
- printf("I am test;hello,编程珠玑 ");
- test1();
- }
- // test.h
- void test();
- //test1.c
- #include <stdio.h>
- #include "test1.h"
- void test1()
- {
- printf("test1,needed by test ");
- }
- // test1.h
- void test1();
分别制作动态库libtest.so和libtest1.so,这在后面的示例中会用到:
- $ gcc test1.c -fPIC -shared -o libtest1.so
- $ gcc test.c -fPIC -shared -o libtest.so -L. -ltest1
这样你在当前目录下就会看到有一个libtest.so和libtest1.so文件生成了,其中litest.so依赖libtest.so
注意,由于libtest.so依赖libtest1.so,这里用-L指定了要链接的test1的路径,因此我们看到:
- $ ldd libtest.so
- linux-vdso.so.1 (0x00007ffd1bbca000)
- libtest1.so => not found
- libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007f9f1d0ae000)
- /lib64/ld-linux-x86-64.so.2 (0x00007f9f1d6a1000)
从这里可以看出libtest是依赖libtest1库的,但是特别注意到,libtest1.so指向的是not found,这会有什么影响吗?我们后面就会看到。
链接时查找路径
我们都知道,在编译成可执行文件前,链接器链接动态库也是需要查找动态库路径的,否则怎么链接上指定的动态库呢?那么这个顺序又是怎样的呢?
首先会查找的会是编译时链接的路径。修改前面的main.c,让它调用libtest.so中的test函数:
- // 来源:公众号【编程珠玑】
- #include <stdio.h>
- #include "test.h"
- int main()
- {
- test(); // 调用libtest.so中的test函数
- return 0;
- }
编译链接:
- $ gcc -o main main.c -I ./ -L./ -ltest -ltest1
完美编译过。除此之外,如果我们把libtest.so和libtest1.so都移到/usr/lib下面,我们发现,即便不用-L也能编译过了:
- $ gcc -o main main.c -I ./ -ltest -ltest1
这里需要说明的是,我们通过-L./来指定搜索库的路径,由于libtest.so依赖libtest1.so,因此在编译链接时,也需要链接上test1。