共享库在程序编译并不会链接到目标代码中,实在程序运行是才被载入。不同的应用程序如果调用相同的库,那么在内存中只需要有一份共享库的实例,规避了空间浪费问题。
动态库在程序运行被载入,解决了静态库对程序的更新、部署和发布会带来麻烦,用户只需要更新动态库即可,增量更新。
动态库文件以.so结尾通常最终的库名称是libxxx.so
动态库制作
步骤一:生成目标文件,添加编译选项:-fpic
gcc -fpic -c add.c
参数:-fpic 创建与地址无关的编译程序
步骤二:生成共享库,此时要加连接器选项:-shared(指定生成动态链接库)
gcc -shared add.o sub.o mul.o div.o -o libtest.so
步骤三:通过nm命令查看对应的函数
nm libtest.so | grep "add"
ldd查看可执行文件的依赖的动态库
ldd a.out
动态库测试
引用动态库编译可执行文件(和静态库的方式是一样)
gcc test.c -L./ -I./ -ltest
但是运行./a.out却报错,找不到对应的库文件,可以将动态库放到标准库(不推荐)里面去。
- 当系统加载可执行代码的时候,必须知道其所依赖的库的名字,还需要知道绝对路径。此时就需要系统动态载入器
- 对于elf格式的可执行程序,是由ld-linux.so*来完成的,它先搜索找到文件的DT_RPATH段---环境变量LD_LIBRARY_PATH --- /etc/id.so.cache文件列表-/lib/ ,/usr/lib目录找到库文件后将其载入内存
如何让系统找到动态库
- 拷贝自己制作的共享库到/lib或者/usr/lib(不能是/lib64)
- 临时设置LD_LIBRARY_PATH
export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/home/gy/test_linux/share_lib2
- 永久设置,把 export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:库路径,设置到~/.bashrc
vim ~/.bashrc
最后一行添加
export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/home/gy/test_linux/share_lib2
设置完最后
source ~/.bashrc
- 将其添加到/etc/ld.so.conf文件中
编辑/etc/ld.so.conf文件,加入库文件所在目录的路径
运行sudo ldconfig -v, 该命令会重建/etc/ld.so.cache文件 - 使用符号链接,但是一定要使用绝对路径
ln -s 库路径绝对路径 /lib/libxxx.so