库
也叫代码库,把一个些目标文件合并在一起方便使用,有静态库和共享库两种。
静态库:在链接时把库的二进制指令复制到调用模块中。
共享库:会和调用者一起加载到内存,当执行调用语句时会从程序的调用位置跳转到共享库中运行。
优缺点:
静态库的优点是运行速度快,但维护麻烦,当静态库中人内容更新后需要重新编译程序,
使用静态库编译出的可执行文件会比共享库的要大。
共享库的优点是使用方便,共享库如果发生变化不需要重新编译程序,
使用它编译出的可执行文件比使用静态库要小,运行速度要比使用静态库慢。
注意:静态库的扩展名是.a,共享库的扩展名是.so,共享库要有执行权限。
静态库的创建与使用
1、创建静态库
编辑静态库原码:vim .c/.h
编译出目标文件:gcc -c xxx.c -> xxx.o
把目标文件打包成静态库文件:ar -r libxxx.a x1.o x2.o ...
ar 是一个专门控制静态库的命令
-r 把目录文件合并成一个静态库,如果静态库文件已经存在则更新。
-q 向静态库中添加目录文件
-t 查看静态库中有哪些目标文件
-d 从静态库中删除目标文件
-x 把静态库展开为目标文件
2、使用静态库
1、直接调用
把共享库当作目标文件一样,与调用者的目标文件一起合并出可执行文件。
gcc main.c libcalc.a
2、通过设置LIBRARY_PATH环境变量来指定库的路径
gcc main.c -lcalc 需要通过-l来指定库名
3、通过gcc -L参数来指定库的路径
gcc main.c -L./calc/ -lclac
共享库的创建与使用
1、创建共享库
编辑静态库原码:vim .c/.h
编译出目标文件:gcc -c -fpic xxx.c -> xxx.o
把目标文件打包成共享库:gcc -shared x1.o x2.o ... -o libxxx.so
-fpic编译出位置无关代码,在代码中使用相对地址,这样共享库就可以遇到内存的任何位置。
2、使用共享库
1、直接调用
gcc main.c libcalc.so
注意:需要设置共享的加载路径,LD_LIBRARY_PATH
2、通过设置LIBRARY_PATH环境变量来指定库的路径
gcc main.c -lcalc 需要通过-l来指定库名
注意:如果静态库和共享库同时存在,优先使用共享库,通过-static可以指定使用静态库。
3、通过gcc -L参数来指定库的路径
gcc main.c -L./calc/ -lclac
4、动态加载共享库
#include <dlfcn.h>
void *dlopen(const char *filename, int flag);
功能:打开共享库
filename:共享库的路径
flag:打开方式
RTLD_LAZY:延迟加载,使用到共享库时再加载
RTLD_NOW:立即加载
返回值:成功返回共享库的句柄,失败返回NULL
void *dlsym(void *handle, const char *symbol);
功能:通过函数名在共享库中获取函数指针
handle:共享库的句柄
symbol:函数名
返回值:函数地址,失败返回NULL
char *dlerror(void);
功能:获取错误信息
int dlclose(void *handle);
功能:卸载共享库
注意:编译时添加-ldl参数
辅助工具
nm:查看目标文件、可执行文件、静态库、共享库文件的符号列表。
ldd:查看可执行文件依赖了哪些共享库
用法:ldd [选项]… 文件…
--help 印出这份说明然后离开
--version 印出版本信息然后离开
-d, --data-relocs 进程数据重寻址
-r, --function-relocs 进程数据和函数重寻址
-u, --unused 印出未使用的直接依赖关系
-v, --verbose 印出所有信息
例如:终端中输入 ldd a.out 打印以下数据
linux-gate.so.1 => (0xb7fc4000)
libc.so.6 => /lib/i386-linux-gnu/libc.so.6 (0xb7df3000)
/lib/ld-linux.so.2 (0xb7fc6000)
strip:删除目标文件、可执行文件、静态库、共享库文件中符号列表、调试信息,可以有效降低文件的大小。
用法:strip <选项> 输入文件
例如:strip a.out
objdump:可以显示二进制文件的汇编信息