静态库
静态库:程序在编译链接的时候把库的代码链接到可执行文件中。程序运行的时候将不再需要静态库。
制作静态库
现在有两个文件:test.h, test.c,如下。
制作静态库的流程如下:
执行上述命令后,结果如下:lib目录里存储制作的静态库。
使用静态库
现在我们创建main.c来测试我们的静态库。如下:
这里出现了一个报错,找不到头文件。原因很简单,test.h与main.c不在同一级目录下,此时我们有3种做法:
- 告诉编译器,除了在当前目录下、系统目录下找,如果找不到,就去指定目录下找。参数
-I [路径】
gcc main.c -o main -I ./lib/include
- include头文件时,带相对路径或绝对路径
include "lib/include/test.h"
- 安装到系统里,即将头文件和库移到
/usr/inlcude 和/usr/lib64
, 如果不想移,也可以建立软链接,再将软链接放到/usr/inlcude 和/usr/lib64
这里采用第一种,结果如下
报错显示找不到aPrint函数,属于链接报错。原因是你只告诉了.h的路径,并没有告诉.a文件的路径.
因此完整的命令如下
gcc main.c -o main -I ./lib/include/ -L ./lib/mylib -ltest
其中-L [库的路径]-l [库名]
注意库名要掐头去尾且最好与-l连在一起
最后运行一下,正常。
动态库
动态库:程序在运行的时候才去链接动态库的代码,多个程序共享使用库的代码。
linux下有个命令可以看到程序动态链接了哪些库 — ldd
以上文的main程序为例
(动态库的文件扩展名通常取为.so)
制作动态库
现在有两个文件:dytest.h, dytest.c,如下。
与制作静态库的流程一样,先生成目标文件,再打包。
执行上述命令后便会生成lib目录,libtest.so便是我们制作的动态库
使用动态库
创建main.c进行测试
同样和静态库一样:gcc main.c -o main -I ./lib/include/ -L ./lib/mylib/ -ltest
最后便会出现下面的错误
此时我们用ldd命令来查看,会看到not found,找不到库,可我们明明已经告诉编译器库在哪里,为什么还是找不到?原因是我们告诉了编译器,但现在已经是可执行程序,需要通过加载器来进行链接,但我们并没有告诉加载器,库在哪里。
解决这个问题的方法很多,这里介绍3种:
- 安装到系统里,即
/usr/lib64
- 添加路径到环境变量
LD_LTBRAY_PATH
export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/home/sfw/study/3_10/lib/mylib
地址写到库所在目录就可以了,因为库的名字早就写在可执行程序中
但是这种方法,在重启时,环境变量更新,便会失效。除非,你写到系统配置文件里。 - 在
/etc/ld.so.conf.d
里面建立动态库路径,然后ldconfig
/etc/ld.so.conf.d 目录是 Linux 下的动态链接器配置目录。在这个目录下,可以放置一系列以 .conf 结尾的文件,这些文件包含了动态链接器的库文件搜索路径配置。每个文件通常包含一组路径,告诉动态链接器在运行时去哪里搜索共享库文件。
操作如下:(下面操作需要root权限)
一般我们网上下载库,最常用的方法是直接安装到系统里。
动态库是如何被加载的
现在有两个可执行程序1.exe 2.exe.它们都链接了动态库dy.so。当程序运行时,dy.so会被加载到内存里,然后通过页表,映射到进程1和进程2的进程地址空间里的共享区。此时进程便可以正常执行。如下图。
小知识:
通过下图:我们知道多个进程是共用一个动态库。也就是下图物理内存中的红色区域。假设dy.so中有一个全局变量,假设为int errno = 0
, 如果进程1中errno变为0了,那在进程2中errno的值会被为0吗?很显然不会,原因是发生了写时拷贝