还在纠结C语言中的动态库和静态库,比较一下二者有何优缺点

嵌入式ARM 2020-02-18 00:00

来源 :https://blog.csdn.net/u010649766/article/details/78528601


函数的重要性


我们在编写一个C语言程序的时候,经常会遇到好多重复或常用的部分,如果每次都重新编写固然是可以的,不过那样会大大降低工作效率,并且影响代码的可读性,更不利于后期的代码维护。我们可以把他们制作成相应的功能函数,使用时直接调用就会很方便,还可以进行后期的功能升级。


例如我要在一段代码中多次交换两个变量的值,我可以在代码中多次写入:


i=x;
x
=y;
y
=i;


不过这样未免有点麻烦我们可以编写一个change_two_int()函数进行简化。


定义如下函数:

void change_two_int( int *a,int *b )

{

int c;

c=*a;

a=b;

*b=c;

}


这样每次要进行交换时只需调用change_two_int(&x , &y); 即可,是否方便了许多?


那么,我们要讨论的和这些有什么关系呢?库通俗的说就是把这些常用函数的目标文件打包在一起,提供相应函数的接口,便于程序员使用。库是别人写好的现有的,成熟的,可以复用的代码,我们只需要知道其接口如何定义,便可以自如使用。


共享库=静态库


现实中每个程序都要依赖很多基础的底层库,不可能每个人的代码都从零开始,因此库的存在意义非同寻常。比如我们常使用的printf函数,就是C标准库提供的函数。我们在使用时只需要包含相应的头文件就可以使用(非静态编译还要有相应的库文件)。而不用关心printf函数具体是如何实现的,这样就大大提高了程序员编写代码的效率。


从使用方法上分库大体上可以分为两类:静态库和共享库。


在windows中静态库是以.lib为后缀的文件,共享库是以.dll为后缀的文件。在linux中静态库是以.a为后缀的文件,共享库是以.so为后缀的文件。


以linux下的静态库和动态库为例我们研究一下,首先我们看一下他们的生成方式。


▶静态库

  • 首先将源文件编译成目标文件:gcc –c a.c b.c

  • 生成静态库:ar –rc libstatic.a a.o b.o


▶共享库

  • 同静态库一样编译成目标文件:gcc –c a.c b.c

  • 生成共享库:gcc –fPIC –shared –o libshared.so a.o b.o


由此可见静态库和动态库都是对目标文件的处理,也可以说库文件已经是机器码文件了,静态库和共享库的加载过程有很大的区别。


  • 静态库的链接方法:

gcc –o staticcode –L. –lstatic main.c –static(默认库在当前文件夹)


  • 共享库的链接方法:

gcc –o sharedcode -L. –lshared main.c(默认库在当前文件夹)


静态库


当程序与静态库连接时,库中目标文件所含的所有将被程序使用的函数的机器码被copy到最终的可执行文件中。这就会导致最终生成的可执行代码量相对变多,相当于编译器将代码补充完整了,优点是这样运行起来相对就快些。


不过会有个缺点:占用磁盘和内存空间。静态库会被添加到和它连接的每个程序中,而且这些程序运行时,都会被加载到内存中。无形中又多消耗了更多的内存空间。


动态库


与共享库连接的可执行文件只包含它需要的函数的引用表,而不是所有的函数代码,只有在程序执行时, 那些需要的函数代码才被拷贝到内存中。


优点:这样就使可执行文件比较小,节省磁盘空间,更进一步,操作系统使用虚拟内存,使得一份共享库驻留在内存中被多个程序使用,也同时节约了内存。


缺点:不过由于运行时要去链接库会花费一定的时间,执行速度相对会慢一些,总的来说静态库是牺牲了空间效率,换取了时间效率,共享库是牺牲了时间效率换取了空间效率,没有好与坏的区别,只看具体需要了。


另外,一个程序编好后,有时需要做一些修改和优化,如果我们要修改的刚好是库函数的话,在接口不变的前提下,使用共享库的程序只需要将共享库重新编译就可以了,而使用静态库的程序则需要将静态库重新编译好后,将程序再重新编译一遍。这也是使用过程当中的差别,以现在的项目举例,在远程更新的时候,如果只是*.so动态库封装内容变化了,那么只需要更新*.so即可。


总结


▶静态库和动态库在两种系统下存在形式


Windows下:

  • .dll动态库

  • .lib静态库

  • 库即为源代码的二进制文件


Linux下:

  • .so动态库

  • .a静态库


▶静态库和动态库的优缺点


静态库在程序编译时会被连接到目标代码中,程序运行时将不再需要该静态库。


动态库在程序编译时并不会被连接到目标代码中,而是在程序运行时才被载入,因此在程序运行时还需要动态库存在。


(1)库文件是如何产生的在linux下?


静态库的后缀是.a,它的产生分两步:

  • 由源文件编译生成一堆.o,每个.o里都包含这个编译单元的符号表。

  • ar命令将很多.o转换成.a,成文静态库。


动态库的后缀是.so,它由gcc加特定参数编译产生。


例如:

gcc−fPIC−c∗.cgcc−fPIC−c∗.c gcc -shared -Wl,-soname, libfoo.so.1 -olibfoo.so.1.0 *.


(2)库文件是如何命名的,有没有什么规范?


在linux下,库文件一般放在/usr/lib和/lib下:

  • 静态库的名字一般为libxxxx.a,其中xxxx是该lib的名称。

  • 动态库的名字一般为libxxxx.so.major.minor,xxxx是该lib的名称,major是主版本号, minor是副版本号。


(3)如何知道一个可执行程序依赖哪些库?


ldd命令可以查看一个可执行程序依赖的共享库。


例如 #ldd /bin/lnlibc.so.6,可以看到ln命令依赖于libc库和ld-linux库:

=> /lib/libc.so.6 (0×40021000)/lib/ld-linux.so.2

=> /lib/ld- linux.so.2 (0×40000000)


(4)可执行程序在执行的时候,如何定位共享库文件?


当系统加载可执行代码时候,能够知道其所依赖的库的名字,但是还需要知道绝对路径。此时就需要系统动态载入器(dynamiclinker/loader)。


对于elf格式的可执行程序,是由ld-linux.so*来完成的。它先后搜索elf文件的DT_RPATH段—环境变量LD_LIBRARY_PATH—/etc/ld.so.cache文件列表—/lib/,/usr/lib目录,找到库文件后将其载入内存。


(5)在新安装一个库之后,如何让系统能够找到它?


如果安装在/lib或者/usr/lib下,那么ld默认能够找到,无需其他操作。


如果安装在其他目录,需要将其添加到/etc/ld.so.cache文件中,步骤如下:

  • 编辑/etc/ld.so.conf文件,加入库文件所在目录的路径。

  • 运行ldconfig,该命令会重建/etc/ld.so.cache文件。


▶linux中编译静态库和动态库的基本方法


(1)静态库


在linux环境中, 使用ar命令创建静态库文件。


以下是命令的选项:

  • d—从指定的静态库文件中删除文件。

  • m—把文件移动到指定的静态库文件中。

  • p—把静态库文件中指定的文件输出到标准输出。

  • q—快速地把文件追加到静态库文件中。

  • r—把文件插入到静态库文件中。

  • t—显示静态库文件中文件的列表。

  • x—从静态库文件中提取文件。


还有多个修饰符修改以上基本选项,详细请man ar。


以下列出三个:

  • a—把新的目标文件(*.o)添加到静态库文件中现有文件之后。

  • b—*****之前。

  • v—使用详细模式。


ar命令的命令行格式如下:

ar[-]{dmpqrtx}[abcfilNoPsSuvV][membername] [count] archive files…


参数archive定义库的名称,files是库文件中包含的目标文件的清单,用空格分隔每个文件。


比如,创建一个静态库文件的命令如下:

ar r libapue.a error.oerrorlog.o lockreg.o


这样就了libapue.a静态库文件,可以用t选项显示包含在库中的文件。


创建库文件之后,可以创建这个静态库文件的索引来帮助提高和库连接的其他程序的编译速度。


使用ranlib程序创建库的索引,索引存放在库文件内部:

ranlib libapue.a


用nm程序显示存档文件的索引,它可以显示目标文件的符号:

nm libapue.a | more


如果是显示目标文件的符号:

nm error.o | more


如何使用呢?如下所示:

gcc -o test test.c libapue.a


这样就可以在test.c中调用在libapue.a中的函数了。


(2)动态库


  • 创建共享库:

gcc -shared -o libapue.soerror.o errorlog.o


  • 编译共享库:


假设共享库位于当前目录(即跟程序文件相同的目录中):

gcc -o test -L. -lapue test.c


这样就编译出了不包含函数代码可执行文件了,但是你在运行时会发现linux动态加载器找不到libapue.so文件。


可以用ldd命令查看可执行文件依赖什么共享库:

ldd test


  • 如何才能让动态加载器发现库文件呢?


有两种方法可以解决:

  • 环境变量。

  • 修改/etc/ld.so.conf文件。


环境变量

exportLD_LIBRARY_PATH=”$LD_LIBRARY_PATH:.”


修改/etc/ld.so.conf文件


一般应用程序的库文件不与系统库文件放在同一个目录下,一般把应用程序的共享库文件放在/usr/local/lib下,新建一个属于自己的目录apue,然后把刚才libapue.so复制过去就行了。


同时在/etc/ld.so.conf中新增一行:

/usr/local/lib/apue


以后在编译程序时加上编译选项:

-L/usr/local/lib/apue -lapue


参数的配置通过mangcc可以看到:

-llibrary


连接名为library的库文件。连接器在标准搜索目录中寻找这个库文件,库文件的真正名字。


-END-




推荐阅读



【01】工程师总结:单片机C语言编程心得
【02】为何强烈不建议中途转IT的人从C语言开始?
【03】超长文详解:C语言预处理命令(精品,建议收藏)
【04】瞬间打通奇经八脉!C语言知识点思维导图!
【05】C语言中最大烦恼被解决!指针和内存泄漏攻坚战


免责声明:整理文章为传播相关技术,版权归原作者所有,如有侵权,请联系删除

嵌入式ARM 关注这个时代最火的嵌入式ARM,你想知道的都在这里。
评论
  • 习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习笔记&记录学习习笔记&记学习学习笔记&记录学习学习笔记&记录学习习笔记&记录学习学习笔记&记录学习学习笔记记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记
    youyeye 2024-12-12 10:13 79浏览
  • 习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习笔记&记录学习习笔记&记学习学习笔记&记录学习学习笔记&记录学习习笔记&记录学习学习笔记&记录学习学习笔记记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记
    youyeye 2024-12-11 17:58 108浏览
  • 一、SAE J1939协议概述SAE J1939协议是由美国汽车工程师协会(SAE,Society of Automotive Engineers)定义的一种用于重型车辆和工业设备中的通信协议,主要应用于车辆和设备之间的实时数据交换。J1939基于CAN(Controller Area Network)总线技术,使用29bit的扩展标识符和扩展数据帧,CAN通信速率为250Kbps,用于车载电子控制单元(ECU)之间的通信和控制。小北同学在之前也对J1939协议做过扫盲科普【科普系列】SAE J
    北汇信息 2024-12-11 15:45 140浏览
  • 时源芯微——RE超标整机定位与解决详细流程一、 初步测量与问题确认使用专业的电磁辐射测量设备,对整机的辐射发射进行精确测量。确认是否存在RE超标问题,并记录超标频段和幅度。二、电缆检查与处理若存在信号电缆:步骤一:拔掉所有信号电缆,仅保留电源线,再次测量整机的辐射发射。若测量合格:判定问题出在信号电缆上,可能是电缆的共模电流导致。逐一连接信号电缆,每次连接后测量,定位具体哪根电缆或接口导致超标。对问题电缆进行处理,如加共模扼流圈、滤波器,或优化电缆布局和屏蔽。重新连接所有电缆,再次测量
    时源芯微 2024-12-11 17:11 141浏览
  • 首先在gitee上打个广告:ad5d2f3b647444a88b6f7f9555fd681f.mp4 · 丙丁先生/香河英茂工作室中国 - Gitee.com丙丁先生 (mr-bingding) - Gitee.com2024年对我来说是充满挑战和机遇的一年。在这一年里,我不仅进行了多个开发板的测评,还尝试了多种不同的项目和技术。今天,我想分享一下这一年的故事,希望能给大家带来一些启发和乐趣。 年初的时候,我开始对各种开发板进行测评。从STM32WBA55CG到瑞萨、平头哥和平海的开发板,我都
    丙丁先生 2024-12-11 20:14 97浏览
  • 应用环境与极具挑战性的测试需求在服务器制造领域里,系统整合测试(System Integration Test;SIT)是确保产品质量和性能的关键步骤。随着服务器系统的复杂性不断提升,包括:多种硬件组件、操作系统、虚拟化平台以及各种应用程序和服务的整合,服务器制造商面临着更有挑战性的测试需求。这些挑战主要体现在以下五个方面:1. 硬件和软件的高度整合:现代服务器通常包括多个处理器、内存模块、储存设备和网络接口。这些硬件组件必须与操作系统及应用软件无缝整合。SIT测试可以帮助制造商确保这些不同组件
    百佳泰测试实验室 2024-12-12 17:45 113浏览
  • 全球智能电视时代来临这年头若是消费者想随意地从各个通路中选购电视时,不难发现目前市场上的产品都已是具有智能联网功能的智能电视了,可以宣告智能电视的普及时代已到临!Google从2021年开始大力推广Google TV(即原Android TV的升级版),其他各大品牌商也都跟进推出搭载Google TV操作系统的机种,除了Google TV外,LG、Samsung、Panasonic等大厂牌也开发出自家的智能电视平台,可以看出各家业者都一致地看好这块大饼。智能电视的Wi-Fi连线怎么消失了?智能电
    百佳泰测试实验室 2024-12-12 17:33 119浏览
  • 本文介绍瑞芯微RK3588主板/开发板Android12系统下,APK签名文件生成方法。触觉智能EVB3588开发板演示,搭载了瑞芯微RK3588芯片,该开发板是核心板加底板设计,音视频接口、通信接口等各类接口一应俱全,可帮助企业提高产品开发效率,缩短上市时间,降低成本和设计风险。工具准备下载Keytool-ImportKeyPair工具在源码:build/target/product/security/系统初始签名文件目录中,将以下三个文件拷贝出来:platform.pem;platform.
    Industio_触觉智能 2024-12-12 10:27 115浏览
  • 铁氧体芯片是一种基于铁氧体磁性材料制成的芯片,在通信、传感器、储能等领域有着广泛的应用。铁氧体磁性材料能够通过外加磁场调控其导电性质和反射性质,因此在信号处理和传感器技术方面有着独特的优势。以下是对半导体划片机在铁氧体划切领域应用的详细阐述: 一、半导体划片机的工作原理与特点半导体划片机是一种使用刀片或通过激光等方式高精度切割被加工物的装置,是半导体后道封测中晶圆切割和WLP切割环节的关键设备。它结合了水气电、空气静压高速主轴、精密机械传动、传感器及自动化控制等先进技术,具有高精度、高
    博捷芯划片机 2024-12-12 09:16 111浏览
  • 在智能化技术快速发展当下,图像数据的采集与处理逐渐成为自动驾驶、工业等领域的一项关键技术。高质量的图像数据采集与算法集成测试都是确保系统性能和可靠性的关键。随着技术的不断进步,对于图像数据的采集、处理和分析的需求日益增长,这不仅要求我们拥有高性能的相机硬件,还要求我们能够高效地集成和测试各种算法。我们探索了一种多源相机数据采集与算法集成测试方案,能够满足不同应用场景下对图像采集和算法测试的多样化需求,确保数据的准确性和算法的有效性。一、相机组成相机一般由镜头(Lens),图像传感器(Image
    康谋 2024-12-12 09:45 117浏览
  • RK3506 是瑞芯微推出的MPU产品,芯片制程为22nm,定位于轻量级、低成本解决方案。该MPU具有低功耗、外设接口丰富、实时性高的特点,适合用多种工商业场景。本文将基于RK3506的设计特点,为大家分析其应用场景。RK3506核心板主要分为三个型号,各型号间的区别如下图:​图 1  RK3506核心板处理器型号场景1:显示HMIRK3506核心板显示接口支持RGB、MIPI、QSPI输出,且支持2D图形加速,轻松运行QT、LVGL等GUI,最快3S内开
    万象奥科 2024-12-11 15:42 123浏览
我要评论
0
点击右上角,分享到朋友圈 我知道啦
请使用浏览器分享功能 我知道啦