本文将演示4种各自独立的得到最终二进制文件的方式。代码采用C语言。
- 用gcc将C语言代码生成静态库 .a 文件,再与编译后的 main.o 合成最终的静态链接的可执行文件,查看运行结果。
- 用gcc将C语言代码生成动态库,待用。
- 用gcc将C语言代码编译并链接动态库,生成可执行文件main,运行时依赖动态库so文件。
- 演示用ar如何转换 静态库文件得到动态库文件。可被用于可执行文件的链接。
本文代码文件内容
首先列出所有代码文件内容,一共3个文件:drive.h,drive.c,main.c,分别为 动态库libdrive.so 的头文件、函数实现文件、主入口main()文件。内容分别如下。为了简明易懂,只以最简单的功能实现。
(1) drive.h 文件内容:
(2) drive.c 文件内容:
(3) main.c 文件内容
用gcc将C语言代码生成静态库 .a,被链接入最终可执行文件
(1) 编译 drive.c 文件为.o 文件:
gcc 的几个重要编译参数,上面用到了-c 、-o 等参数,下面还会用到所以再次贴上参数说明:
(2) 使用编译drive.c得到的drive.o 文件作为材料,生成 libdrive.a 静态库文件。文件名前面加lib是为了gcc链接文件时 默认的约定找以lib开始的库文件:
(3) 链接得到最终可执行文件:
当前目录下只有libdrive.a 库,没有动态库,所以不会优先链接.so库,只能链接libdrive.a库。所以我使用file 命令查看main-static的成分,我以为它就是静态文件,结果发现怎么还是动态链接的?(链接了3个动态库文件)
虽然通过-ldrive 我们确实链接了libdrive.a静态文件进入 main-static文件内部(所以以上出现的链接信息里不会显示libdrive.so 或 libdrive.a,但file的结果说得到的main-static文件仍然是dynamically linked得的,为什么会这样呢?
新安装的libc的静态库文件的路径:
重新尝试实现彻底的静态链接:
以上 ldd 的结果说明,main-static-true 文件才是真正静态链接的。
来对比一下 假的 main-static 跟真的可执行文件大小差距多大:
文件大小相差45倍。由于功能少所以这个倍数并不严谨。但大体上我们可以了解纯静态的可执行文件,确实会比 dynamic linked的可执行文件大很多。不过由于现在硬盘和内存的扩大,成本降低,越来越多新编程语言默认采用纯静态编译可执行文件。以增强跨平台的运行能力。
用gcc将 drive.c 代码生成动态库文件 libdrive.so
将当前目录下的所有.o文件(目前只有 drive.o一个文件)作为材料,生成最终的 libdrive.so动态库文件(实际相当于将 .o文件打了个压缩包到 .a文件里,并调整了链接符号)。
所以我们目录下截至现在已有这些文件:
既然是动态库,总要能被调用者直到你的暴露的借口。我们验证下这个libdrive.so动态库是否也暴露了add函数呢?
从结果看,确实暴露了add 函数。至于其他4个W是什么作用,我们目前暂时不管。
用gcc将C语言 main.c 编译并链接动态库,生成可执行文件 main_share ,运行时是否依赖动态库so文件?
上面说过,当libdrive.so 不存在时,libdrive.a 存在,则gcc 根据链接 -L./ -ldrive 参数,在对应目录下只找libdrive.a,所以以前只能链接.a文件。现在有了.so文件,那么-L./ -ldrive会优先链接libdrive.so。让我们用ldd 命令检验一下:
这里我们在命令中临时设置 库文件搜索路径包含当前路径$PWD。
从结果看,我们的main_share 确实链接了动态库 libdrive.so。
「完」
演示用ar如何转换 静态库文件得到动态库文件。可被用于可执行文件的链接。
先清理一下现场:删掉 drive.o libdrive.so 这两个文件。
这里使用现有的 libdrive.a 动态库,看能否转化成 libdrive.so 文件。
(1) 之前说过.a相当于.o 文件的压缩包,那么我们可以分离出 drive.o 文件
(2) 编译生成 .so 文件
(3) 查看导出函数
结果与上上面的 nm -D libdrive.so 输出一致。作为验证我们将使用这个转化出的so动态库测试对main()的链接和运行效果:
警告信息不管。运行效果符合预期,打印了3+5的结果。说明静态库转成动态库也是OK可以用的(实际场景gcc还要加-fPIC 参数来生成动态库,以增强动态库的通用性不依赖于固定地址,本文不深入解析 -fPIC,请读者自己查信息 )。
总结一下,经过以上4个场景的演示验证,我们对Linux的动态库、静态库之间的关系,有了更深入的认知。