- 笔记搬运工
一、库的科普
1、什么是库文件?
- 库是二进制文件,是源代码文件的另一种表现形式,是一些功能相近或相似的函数集合体;
2、使用库的好处
- 提高代码可重用性,提高程序的健壮性(优秀的库往往是资深程序员编写的,久经考验)
- 可减少开发者的开发代码量,缩短开发周期;
3、库制作完成后, 如何给用户使用
- 头文件—包含了库函数的声明
- 库文件—包含了库函数的代码实现
- 注意: 库不能单独使用, 只能作为其他执行程序的一部分完成某些功能, 也 就是说只能被其他程序调用才能使用.
库可分静态库(static library)和共享库(shared library)
二、静态库static library
2.1、静态库可以认为是一些目标代码的集合
在可执行程序运行前就已经加入到执行码中, 成为执行程序的一部分. 按照习惯, 一般以.a做为文件后缀名.
静态库的命名一般分为三个部分:
- 前缀:lib
- 库名称:自定义即可, 如test
- 后缀 .a
最终静态库的名字为:libtest.a
2.2、静态库的制作
下面以fun1.c , fun2.c和head.h三个文件为例讲述静态库的制作和使用, 其中head.h文件中有函数的声明, fun1.c和fun2.c中有函数的实现.
步骤1:将c源文件生成对应的.o文件
gcc -c fun1.c fun2.c
或者分别生成.o文件:
gcc -c fun1.c -o fun1.o
gcc -c fun2.c -o fun2.o
步骤2:使用打包工具ar将准备好的.o文件打包为.a文件
在使用ar工具是时候需要添加参数rcs
r更新、c创建、s建立索引
命令:ar rcs 静态库名 .o文件
ar rcs libtest1.a fun1.o fun2.o
2.3、静态库的使用
gcc -o main1 main.c -L./ -ltest1 -I./
//静态库:libtest.a,在gcc中使用时要把前后缀去掉,同时源文件要包含对应库的头文件;
2.4、静态库的优缺点
优点:
- 函数库最终被打包到应用程序中,实现是函数本地化,寻址方便、速度快。(库函数调用效率==自定义函数使用效率);
- 程序在运行时与函数库再无瓜葛,移植方便。
缺点:
- 消耗系统资源较大, 每个进程使用静态库都要复制一份, 无端浪费内存。
- 静态库会给程序的更新、部署和发布带来麻烦。如果静态库libxxx.a更新了,所有使用它的应用程序都需要重新编译、发布给用户(对于玩家来说,可能是一个很小的改动,却导致整个程序重新下载);
三、共享库(动态库)
3.1、定义和用法
共享库在编译时不会被连接到目标代码中,而是在程序运行时才被载入;不同的应用程序调用相同的库,那么只需要在内存放一份共享库的拷贝即可,规避了空间浪费问题;动态库在程序运行时载入,解决了静态库对程序的更新、部署和发布带来的麻烦;(即可做到增量更新,类似每天的王者荣耀的不退出更新)
一般以 .so 为后缀,命名分为三部分:前缀:lib ;库名:NAME自己定义 ; 后缀:.so; 如 libNAME.so
![在这里插入图片描述](https://img-blog.csdnimg.cn/572b40d9a81d43b18a3832ad91bc82d2.png?x-oss-process=image/watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBA6YCu6Jm-5oi36IKJ5Lid,size_20,color_FFFFFF,t_70,g_se,x_16)
3.2、共享库的制作
1)生成目标文件.o,此时要加上编译选项:-fPIC (fpic)
gcc -fpic -c func1.c func2.c //参数-fpic创建与地址无关的编译程序,实现多个应用程序共享;
2)生成共享库,加上编译选项:-shared
gcc -shared func1.o func2.o -o libtest2.so
3.3、如何保证自己生成的共享库能让系统找到?
3.4、共享库的特点
- 动态库把对一些库函数的链接载入推迟到程序运行的时期
- 可以实现进程间的资源共享
- 将一些程序升级变得简单
- 做到链接载入我去由程序员在代码中控制
四、静态库和动态库的优缺点
静态库的优点:
1 执行速度快, 是因为静态库已经编译到可执行文件内部了
2 移植方便, 不依赖域其他的库文件
缺点:
1 耗费内存, 是由于每一个静态库的可执行程序都会加载一次
2 部署更新麻烦, 因为静态库修改以后所有的调用到这个静态库的可执行文
件都需要重新编译
动态库的优点:
1 节省内存
2 部署升级更新方便, 只需替换动态库即可, 然后再重启服务.
缺点:
1 加载速度比静态库慢
2 移植性差, 需要把所有用到的动态库都移植.
由于由静态库生成的可执行文件是把静态库加载到了其内部, 所以静态库生成的可执行文件一般会比动态库大.