1.什么是库
在windows平台和linux平台下都大量存在着库。
本质上来说库是一种可执行代码的二进制形式,可以被操作系统载入内存执行。
由于windows和linux的本质不同,因此二者库的二进制是不兼容的。
本文仅限于介绍linux下的库。
2.库的种类
linux下的库有两种:静态库和共享库(动态库)。
二者的不同点在于代码被载入的时刻不同。
静态库的代码在编译过程中已经被载入可执行程序,因此体积较大。
共享库的代码是在可执行程序运行时才载入内存的,在编译过程中仅简单的引用,因此代码体积较小。
3.库存在的意义
库是别人写好的现有的,成熟的,可以复用的代码,你可以使用但要记得遵守许可协议。
现实中每个程序都要依赖很多基础的底层库,不可能每个人的代码都从零开始,因此库的存在意义非同寻常。
共享库的好处是,不同的应用程序如果调用相同的库,那么在内存里只需要有一份该共享库的实例。
4.库文件是如何产生的在linux下
静态库的后缀是.a,它的产生分两步
Step 1.由源文件编译生成一堆.o,每个.o里都包含这个编译单元的符号表
Step 2.ar命令将很多.o转换成.a,成文静态库
动态库的后缀是.so,它由gcc加特定参数编译产生。
例如:
$ gcc -fPIC -c *.c $ gcc -shared -Wl,-soname, libfoo.so.1 -o libfoo.so.1.0 *.
5.库文件是如何命名的,有没有什么规范
在linux下,库文件一般放在/usr/lib /lib下,
静态库的名字一般为libxxxx.a,其中xxxx是该lib的名称
动态库的名字一般为libxxxx.so.major.minor,xxxx是该lib的名称,major是主版本号, minor是副版本号
6.如何知道一个可执行程序依赖哪些库
ldd命令可以查看一个可执行程序依赖的共享库,
例如# ldd /bin/lnlibc.so.6
=> /lib/libc.so.6 (0×40021000)/lib/ld-linux.so.2
=> /lib/ld- linux.so.2 (0×40000000)
可以看到ln命令依赖于libc库和ld-linux库
7.可执行程序在执行的时候如何定位共享库文件
当系统加载可执行代码时候,能够知道其所依赖的库的名字,但是还需要知道绝对路径
此时就需要系统动态载入器(dynamic linker/loader)
对于elf格式的可执行程序,是由ld-linux.so*来完成的,它先后搜索elf文件的 DT_RPATH段—环境变量LD_LIBRARY_PATH—/etc/ld.so.cache文件列表—/lib/,/usr/lib目录找到库文件后将其载入内存
一 库操作工具
1 nm命令列出编入目标文件或二进制文件的所有符号3 共享库使得代码维护的工作大大简化
编译器默认会找的目录可以用 gcc -print-search-dirs 查看
其中libraries就是库文件的搜索路径列表,各路径之间用:号隔开。
objdump -d main //通过反汇编查看main函数
使用静态库的好处:连接器可以从静态库中只取出需要的部分来做链接
gcc -c -fPIC *****.c ****.c ****.c
-f后面跟一些编译选项,PIC是其中一种,表示声称位置无关代码。
objdump -ds把反汇编指令和源代码穿插起来分析
readelf
所以共享库各段的加载地址并没有定死,可以加载到任意位置
gcc main.c -g -L. -lstack -Istack -o main
动态库的链接地址是由动态连接器在做动态链接时搜索到的,共享库的搜索路径由动态连接器决定。ld.so(8)的搜索路径:
1 首先在环境变量LD_LIBRARY_PATH所记录的路径中查找
2 从缓存文件/etc/ld.so.cache中查找。这个缓存文件由ldconfig命令读取配置文件/etc/ld.so.conf之后生成
可以将库文件所在的绝对路径添加到/etc/ld.so.conf中。
3 默认路径 /usr/lib /lib
1 gcc -fPIC -g -c liberr.c -o liberr.o
gcc -g -shared -Wl,-soname,liberr.so -o liberr.so.1.0.0 liberr.o -lc
ln -s liberr.so.1.0.0 liberr.so.1
ln -s liberr.so.1.0.0 liberr.so
gcc -g errtest.c -o errtest -L. -lerr
$LD_LIBRARY_PATH=$(pwd) ./errtest
或者
export LD_LIBRARY_PATH=$(pwd)
./errtest
按照共享库的命名惯例,每个共享库有三个文件名:real name、soname、linker name。
真正的库文件的名字是real name,包含完整的共享库版本号。
soname是一个符号链接的名字,只包含共享库的主版本号,引用程序的.dynamic断只记录共享库的soname
linker name仅在编译链接时使用,gcc的-L选项应该指向linker name所在的目录。
重新编译libstack,指向它的soname
gcc -shared -WL,-soname,libstack.so.1 -o libstack.so.1.0 stack.o push.o pop.o is_empty.o
这样编译生成的libstack.so.1.0是real name,但这个库文件记录了它的soname是libstack.so.1。
如果把libstack.so.1所在的目录加入到/etc/ld.so.conf中,然后运行ldconfig命令,ldconfig会自动创建一个soname的符号链接。
如果想在编译的时候使用共享库,应该再创建一个linker name链接
ln -s libstack.so.1.0 libstack.so