什么是库
1、库文件是计算机上的一类文件,可以简单的把库文件看成一种代码仓库,它提供给使用者一些可以直接拿来用的变量、函数或类。
2、库是特殊的一种程序,编写库的程序和编写一般的程序区别不大,只是库不能单独运行。
3、库文件有两种,静态库和动态库(共享库),区别是:静态库在程序的链接阶段被复制到了程序中;动态库在链接阶段没有被复制到程序中,而是程序在运行时由系统动态加载到内存中供程序调用。
4、库的好处:
1)代码保密
2)方便部署和分发
工作原理
静态库:GCC 进行链接时(静态链接方式),会把静态库中代码打包到可执行程序中
动态库:GCC 进行链接时(动态链接方式),动态库的代码不会被打包到可执行程序中,程序启动之后,动态库会被动态加载到内存中。通过 ldd (list dynamic dependencies)命令可检查动态库依赖关系。
静态库的制作和使用
命名规则:
Linux : libxxx.a
lib : 前缀(固定)
xxx : 库的名字,自己起
.a : 后缀(固定)
Windows : libxxx.lib
静态库的制作:
gcc 获得 .o 文件: gcc -c *.c
将 .o 文件打包,使用 ar 工具(archive)
ar rcs libxxx.a xxx.o xxx.o
r – 将文件插入备存文件中
c – 建立备存文件
s – 索引
静态库的使用
gcc [main.c] -o [目标文件] -I [指定使用的库] -l [静态库的名称] -L [静态库的路径]
动态库的制作和使用
命名规则
Linux : libxxx.so
lib : 前缀(固定)
xxx : 库的名字,自己起
.so : 后缀(固定)
在Linux下是一个可执行文件
Windows : libxxx.dll
动态库的制作
gcc生成.o文件,得到和位置无关的代码:
gcc -c –fpic/-fPIC a.c b.c
gcc得到动态库:
gcc -shared a.o b.o -o libxxx.so
动态库的使用
gcc [main.c] -o [目标文件] -I [指定使用的库] -l [动态库的名称] -L [动态库的路径]
动态库加载失败:
使用ldd main(可执行文件)
查看libxxx.so的依赖关系,如果是libxxx.so=>not found
就需要采用以下方式解决:
如何定位共享库文件呢?
当系统加载可执行代码时候,能够知道其所依赖的库的名字,但是还需要知道绝对路径。此时就需要系统的动态载入器来获取该绝对路径。
法一:添加环境变量
临时
export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:动态库所在的绝对路径
添加后查看是否添加成功:
echo $LD_LIBRARY_PATH
此时使用ldd命令查看依赖关系为:libxxx.so=> 动态库绝对路径/libxxx.so(动态库)
注意:该方法是临时的,只能在当前终端使用,关闭当前终端后就会失效。
永久配置,用户级别
进入home目录,修改.bashrc文件:
vim .bashrc
最后一行加入:
export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:动态库所在的绝对路径
保存并退出:wq
让文件生效执行:
source .bashrc 或者 . .bashrc
再去查看依赖关系就显示能查找到
永久配置,系统级别
sudo vim /etc/profile
最后一行加入:
export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:动态库所在的绝对路径
保存并退出:wq
source /etc/profile
关闭终端,重新打开才可以
再去查看依赖关系就显示能查找到
法二:修改/etc/ld.so.cache文件列表
vim /etc/ld.so.cache 里面都是二进制数据,不能直接修改
采用间接修改:
sudo vim /etc/ld.so.conf
最后一行加入:
动态库所在的绝对路径
保存并退出
更新:
sudo ldconfig
再去查看依赖关系就显示能查找到
法三:不推荐
将生成的动态库文件放入/lib/,/usr/lib目录。
但是这两个目录中有许多系统的库文件,可能会发生未知的错误。
静态库优缺点
优点:
静态库被打包到应用程序中加载速度快
发布程序无需提供静态库,移植方便
缺点
消耗系统资源,浪费内存(重复加载)
更新、部署、发布麻烦
动态库优缺点
优点
可以实现进程间资源共享(共享库)
更新、部署、发布简单
可以控制何时加载动态库
缺点
加载速度比静态库慢
发布程序时需要提供依赖的动态库
一般库比较小使用静态库,比较大用动态库