一、动态库的编译
下面经过一个例子来介绍如何生成一个动态库。这里有一个头文件:so_test.h,三个.c文件:test_a.c、test_b.c、test_c.c,咱们将这几个文件编译成一个动态库:libtest.so。
//so_test.h:
#include "stdio.h"
void test_a();
void test_b();
void test_c();
//test_a.c:
#include "so_test.h"
void test_a()
{
printf("this is in test_a...\n");
}
//test_b.c:
#include "so_test.h"
void test_b()
{
printf("this is in test_b...\n");
}
//test_c.c:
#include "so_test.h"
void test_c()
{
printf("this is in test_c...\n");
}
将这几个文件编译成一个动态库:libtest.so
$ gcc test_a.c test_b.c test_c.c -fPIC -shared -o libtest.so
二、动态库的连接
在一、中,咱们已经成功生成了一个本身的动态连接库libtest.so,下面咱们经过一个程序来调用这个库里的函数。程序的源文件为:test.c。
test.c:
#include "so_test.h"
int main()
{
test_a();
test_b();
test_c();
return 0;
}
将test.c与动态库libtest.so连接生成执行文件test:
$ gcc test.c -L. -ltest -o test
测试是否动态链接,若是列出libtest.so,那么应该是链接正常了
$ ldd test
执行test,能够看到它是如何调用动态库中的函数的。
三、编译参数解析
最主要的是GCC命令行的一个选项:
-shared该选项指定生成动态链接库(让链接器生成T类型的导出符号表,有时候也生成弱链接W类型的导出符号),不用该标志外部程序没法链接。至关于一个可执行文件
-fPIC:表示编译为位置独立的代码,不用此选项的话编译后的代码是位置相关的因此动态载入时是经过代码拷贝的方式来知足不一样进程的须要,而不能达到真正代码段共享的目的。
-L.:表示要链接的库在当前目录中
-ltest:编译器查找动态链接库时有隐含的命名规则,即在给出的名字前面加上lib,后面加上.so来肯定库的名称
LD_LIBRARY_PATH:这个环境变量指示动态链接器能够装载动态库的路径。
固然若是有root权限的话,能够修改/etc/ld.so.conf文件,而后调用 /sbin/ldconfig来达到一样的目的,不过若是没有root权限,那么只能采用输出LD_LIBRARY_PATH的方法了。
四、注意
调用动态库的时候有几个问题会常常碰到,有时,明明已经将库的头文件所在目录 经过 “-I” include进来了,库所在文件经过 “-L”参数引导,并指定了“-l”的库名,但经过ldd命令察看时,就是死活找不到你指定连接的so文件,这时你要做的就是经过修改 LD_LIBRARY_PATH或者/etc/ld.so.conf文件来指定动态库的目录。一般这样作就能够解决库没法连接的问题了。
在linux下能够用export命令来设置这个值,在linux终端下输入:
export LD_LIBRARY_PATH=/opt/au1200_rm/build_tools/bin: $LD_LIBRARY_PATH:
而后再输入:export
即会显示是否设置正确
export方式在重启后失效,因此也能够用 vim /etc/bashrc ,修改其中的LD_LIBRARY_PATH变量。
例如:LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/opt/au1200_rm/build_tools/bin。linux