目录
静态库在程序编译时会被连接到目标代码中,程序运行时将不再需要该静态库。编译之后程序文件大,但加载快,隔离性也好。
动态库在程序编译时并不会被连接到目标代码中,而是在程序运行是才被载入,因此在程序运行时还需要动态库存在。多个应用程序可以使用同一个动态库,启动多个应用程序的时候,只需要将动态库加载到内存一次即可。
编译动态库:
-shared 该选项指定生成动态连接库(让连接器生成T类型的导出符号表,有时候也生成弱连接W类型的导出符号),不用该标志外部程序无法连接。相当于一个可执行文件 -fPIC:表示编译为位置独立的代码,不用此选项的话编译后的代码是位置相关的所以动态载入时是通过代码拷贝的方式来满足不同进程的需要,而不能达到真正代码段共享的目的。 -L.:表示要连接的库在当前目录中 -ltest:编译器查找动态连接库时有隐含的命名规则,即在给出的名字前面加上lib,后面加上.so来确定库的名称
配置库位置: LD_LIBRARY_PATH:这个环境变量指示动态连接器可以装载动态库的路径。 当然如果有root权限的话,可以修改/etc/ld.so.conf文件,然后调用 /sbin/ldconfig来达到同样的目的,不过如果没有root权限,那么只能采用输出LD_LIBRARY_PATH的方法了。
ldd命令可以查看可执行文件依赖的库文件。
1. 测试库的程序文件
1.1 测试库的头文件test.h
//test.h
#ifndef __LIBTEST_H
#define __LIBTEST_H
void myprintf();
int myadd(int a,int b);
#endif
1.2 测试库的源文件test.c
//test.c
#include <stdio.h>
#include "test.h"
void myprintf(){
printf("this is library test func!\r\n");
}
int myadd(int a,int b){
return (a+b);
}
1.3 测试应用程序源文件 main.c
//main.c
#include <stdio.h>
#include "test.h"
int main()
{
myprintf();
int a,b,c;
a=5;
b=10;
c = myadd(a,b);
printf("c=%d\n",c);
return 0;
}
2. 静态库的编译流程
2.1 生成 test.o 的链接文件
gcc -c test.c
2.2 生成 libtest.a 的静态库文件
ar -r libtest.a test.o
2.3 生成 main 的可执行文件
gcc main.c -ltest -L. -o main
2.4 测试 main 运行结果
./main
3. 动态库的编译流程
3.1 生成 test.o 的链接文件
gcc -fPIC -c test.c
3.2 生成 libtest.so 的动态库文件
gcc -shared -o libtest.so test.o
3.3 配置动态库路径的环境变量
export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:$(pwd)
echo $LD_LIBRARY_PATH
3.4 生成 main 的可执行文件
gcc main.c -ltest -L. -o main
3.5 测试 main 运行结果
./main
附录:
在linux下可以用export命令来设置这个值,比如
在linux终端下输入:export LD_LIBRARY_PATH=/usr/local/lib: $LD_LIBRARY_PATH:
然后再输入:export
即会显示是否设置正确
export方式在重启后会失效,所以也可以用 vim /etc/bashrc ,修改其中的LD_LIBRARY_PATH变量。
例如:LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/usr/local/lib。
或者:
可以在 ~/.bashrc 或者 ~/.bash_profile 中加入 export 语句,前者在每次登陆和每次打开 shell 都读取一次,后者只在登陆时读取一次。我的习惯是加到 ~/.bashrc 中,在该文件的未尾,可采用如下语句来使设置生效:
export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/usr/local/lib
修改完后,记得关掉当前终端并重新打开一个新的终端,从而使上面的配置生效。
参考文章:
Linux下动态库(.so)和静态库(.a) 的区别
嵌入式——Linux: 库(静态库,动态库)
linux-export