Linux 静态库和共享(动态)库的创建与使用详解
文章目录
- Linux 静态库和共享(动态)库
- 库的介绍
- 使用库有什么好处
- 库制作完成后, 如何给用户使用
- 静态库(static library)
- 静态库的制作
- ar工具创建lib过程
- 静态库的使用
- 源代码
- 静态库的优缺点
- 优点:
- 缺点:
- 共享库(shared library)/动态库
- 共享库的制作
- 共享库的使用
- 那么如何安装共享库?
- 共享库的特点
- 比较静态库和动态库的优缺点
- 静态库
- 动态库
Linux 静态库和共享(动态)库
库的介绍
库是二进制文件, 是源代码文件的另一种表现形式, 是加了密的源代码;
是一些功能相近或者是相似的函数的集合体.
使用库有什么好处
- 提高代码的可重用性, 而且还可以提高程序的健壮性;
- 可以减少开发者的代码开发量, 缩短开发周期.
库制作完成后, 如何给用户使用
- 头文件—包含了库函数的声明
- 库文件—包含了库函数的代码实现
注意: 库不能单独使用, 只能作为其他执行程序的一部分完成某些功能, 也就是说只能被其他程序调用才能使用.
库可分静态库(static library)和共享/动态库(shared library)。
首先先介绍静态库(static library)
静态库(static library)
静态库可以认为是一些目标代码的集合, 是在可执行程序运行前就已经加入到执行码中, 成为执行程序的一部分. 按照习惯, 一般以.a做为文件后缀名.
静态库的命名一般分为三个部分:
- 前缀:
lib
- 库名称:自定义即可, 如
test
- 后缀:
.a
所以最终的静态库的名字应该为:libtest.a
静态库的制作
下面以fun1.c
,fun2.c
和head.h
三个文件为例讲述静态库的制作和使用, 其中head.h
文件中有函数的声明, fun1.c
和fun2.c
中有函数的实现.
步骤1:将c源文件生成对应的.o文件
gcc -c func1.c func2.c
或者分别生成.o文件:
gcc -c func1.c -o func1.o
gcc -c func2.c -o func2.o
步骤2:使用打包工具ar
将准备好的.o文件打包为.a文件
在使用ar工具是时候需要添加参数rcs
- r更新、c创建、s建立索引
- 命令:
ar rcs 静态库名 .o文件
下图为ar
的帮助文档
ar rcs libtest1.a func1.o func2.o
ar工具创建lib过程
静态库的使用
静态库制作完成之后, 需要将.a文件和头文件一定发布给用户.
假设测试文件为main.c
, 静态库文件为libtest1.a
, 头文件为head.h
用到的参数:
L
:指定要连接的库的所在目录l
:指定链接时需要的静态库, 去掉前缀和后缀I
: 指定main.c文件用到的头文件head.h所在的路径
gcc -o main1 main.c -L./ -ltest1 -I./
运行main1,结果如下图:
源代码
func1.c
/*************************************************************************> File Name: func1.c> Author: 杨永利> Mail: 1795018360@qq.com > Created Time: 2021年12月20日 星期一 09时59分47秒************************************************************************/int add(int a,int b)
{return a+b;
}
func2.c
/*************************************************************************> File Name: func2.c> Author: 杨永利> Mail: 1795018360@qq.com > Created Time: 2021年12月20日 星期一 10时00分17秒************************************************************************/int del(int a,int b){return a-b;}
main.c
/*************************************************************************> File Name: main.c> Author: 杨永利> Mail: 1795018360@qq.com > Created Time: 2021年12月20日 星期一 10时01分15秒************************************************************************/#include <stdio.h>int add(int a,int b);
int del(int a,int b);int main(int argc, char* argv[]){printf("调用库函数add:1+2=%d\n",add(1,2));printf("调用库函数del:4-3=%d\n",add(4,3));return 0;
}
静态库的优缺点
优点:
- 函数库最终被打包到应用程序中,实现是函数本地化,寻址方便、速度快。
(库函数调用效率==自定义函数使用效率) - 程序在运行时与函数库再无瓜葛,移植方便。
缺点:
- 消耗系统资源较大, 每个进程使用静态库都要复制一份, 无端浪费内存。
- 静态库会给程序的更新、部署和发布带来麻烦。如果静态库libxxx.a更新了,所有使用它的应用程序都需要重新编译、发布给用户(对于玩家来说,可能是一个很小的改动,却导致整个程序重新下载)。
共享库(shared library)/动态库
共享库在程序编译时并不会被连接到目标代码中,而是在程序运行是才被载入。不同的应用程序如果调用相同的库,那么在内存里只需要有一份该共享库的拷贝,规避了空间浪费问题。
动态库在程序运行时才被载入, 也解决了静态库对程序的更新、部署和发布会带来麻烦。 用户只需要更新动态库即可,增量更新。
按照习惯, 一般以”.so
”做为文件后缀名. 共享库的命名一般分为三个部分:
- 前缀:
lib
- 库名称:自己定义即可, 如
test
- 后缀:
.so
所以最终的静态库的名字应该为:libtest.so
共享库的制作
步骤1:生成目标文件.o, 此时要加编译选项:-fPIC(fpic)
gcc -fpic -c func1.c func2.c
参数:
-fpic
:创建与地址无关的编译程序(pic, position independent code), 目的就是为了能够在多个应用程序间共享.
步骤2:生成共享库, 此时要加链接器选项: -shared
(指定生成动态链接库)
gcc -shared func1.o func2.o -o libtest2.so
共享库的使用
引用动态库编译成可执行文件(跟静态库方式一样):
用到的参数:
-L
:指定要连接的库的所在目录-l
:指定链接时需要的动态库, 去掉前缀和后缀-I
: 指定main.c文件用到的头文件head.h所在的路径
gcc main.c -I./ -L./ -ltest2 -o main2
然后运行:./main2,发现竟然报错了,如下图,显示找不到libtest2.so库
分析为什么在执行的时候找不到libtest2.so库?
当系统加载可执行代码时候, 能够知道其所依赖的库的名字, 但是还需要知道所依赖的库的绝对路径。此时就需要系统动态载入器(dynamic linker/loader)。
ldd命令
可以查看可执行文件依赖的库文件, 执行ldd main2
, 可以发现libtest2.so找不到.
对于
elf格式
的可执行程序,是由ld-linux.so*
来完成的, 它先后搜索elf文件
的DT_RPATH段 — 环境变量LD_LIBRARY_PATH — /etc/ld.so.cache文件列表 — /lib/, /usr/lib目录
找到库文件后将其载入内存。使用file命令可以查看文件的类型:
file main2
,显示如下图,该可执行文件为,elf格式。
所以说,一旦你定义了一个共享函数库,你还需要安装它。
那么如何安装共享库?
步骤1:拷贝自己制作的共享库到/lib
或者/usr/lib
例如上面案例执行下面语句复制到/usr/lib
sudo cp ~/C/sharelib/libtest2.so /usr/lib
然后在终端运行
sudo ldconfig
步骤2:临时设置LD_LIBRARY_PATH
:
export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:库路径
export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/home/yang/C/sharelib/libtest2.so
永久设置, 把export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:库路径
, 设置到~/.bashrc文件
中(编辑~/.bashrc,将上面设置语句加到最后一行), 然后在执行下列三种办法之一:
- 执行. ~/.bashrc使配置文件生效(第一个.后面有一个空格)
- 执行source ~/.bashrc配置文件生效
- 退出当前终端, 然后再次登陆也可以使配置文件生效
永久设置,把export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:库路径
,设置到/etc/profile
文件中(编辑/etc/profile,将上面设置语句加到最后一行)
步骤3:将其添加到 /etc/ld.so.cache
文件中
编辑/etc/ld.so.conf
文件, 加入库文件所在目录的路径
运行sudo ldconfig -v
, 该命令会重建/etc/ld.so.cache
文件
解决了库的路径问题之后, 再次ldd命令可以查看可执行文件依赖的库文件, ldd main2,libtest.so就能找到了
再编译运行,运行成功,如下图:
共享库的特点
- 动态库把对一些库函数的链接载入推迟到程序运行的时期。
- 可以实现进程之间的资源共享。(因此动态库也称为共享库)
- 将一些程序升级变得简单。
- 甚至可以真正做到链接载入完全由程序员在程序代码中控制(显示调用)
比较静态库和动态库的优缺点
静态库
优点:
- 执行速度快, 是因为静态库已经编译到可执行文件内部了
- 移植方便, 不依赖域其他的库文件
缺点:
- 耗费内存, 是由于每一个静态库的可执行程序都会加载一次
- 部署更新麻烦, 因为静态库修改以后所有的调用到这个静态库的可执行文件都需要重新编译
动态库
优点:
- 节省内存
- 部署升级更新方便, 只需替换动态库即可, 然后再重启服务.
缺点:
- 加载速度比静态库慢
- 移植性差, 需要把所有用到的动态库都移植.
由于由静态库生成的可执行文件是把静态库加载到了其内部, 所以静态库生成的可执行文件一般会比动态库大.
Linux 静态库和共享(动态)库的创建与使用详解相关推荐
- 在Linux中创建静态库.a和动态库.so
转自:http://www.cnblogs.com/laojie4321/archive/2012/03/28/2421056.html 在Linux中创建静态库.a和动态库.so 我们通常把一些公用 ...
- Linux静态库.a与动态库.so的生成与区别、以及.so库文件的封装与使用
#一.前言 如果有公司需要使用你们产品的一部分功能(通过代码调用这些功能),如果不想提供源代码,那么就可以通过封装成库文件的形式提供给对方使用.本文主要介绍了生成动态库与静态库文件的过程.以及封装和使 ...
- [转]Linux下g++编译与使用静态库(.a)和动态库(.os) (+修正与解释)
在windows环境下,我们通常在IDE如VS的工程中开发C++项目,对于生成和使用静态库(*.lib)与动态库(*.dll)可能都已经比较熟悉,但是,在linux环境下,则是另一套模式,对应的静态库 ...
- linux库引入之动态库静态库(生成和使用)
库: 库是一种可执行代码的二进制形式,可以被操作系统载入内存执行.就是将源代码转化为二进制格式的源代码,相当于进行了加密,别人可以使用库,但是看不到库中的内容. 如何使用 用户需要同时具有头文件和库. ...
- 库文件、静态库(lib)与动态库(dll)的生成与使用
静态库 程序编译一般需经预处理.编译.汇编和链接几个步骤.在应用中,有一些公共代码是需要反复使用,就把这些代码编译为"库"文件:在链接步骤中,连接器将从库文件取得所需的代码,复制到 ...
- C++中.lib静态库、.dll动态库的生成及调用1
一.前言 1.动态链接库(dll)与静态链接库(lib): 动态链接库(dll)是一个可以被其他应用程序共享的程序模块,其中封装了一些可以被共享的例程和资源.在链接步骤中,连接器将从库文件取得所需的代 ...
- 【Android NDK 开发】Ubuntu 函数库交叉编译 ( Android 动态库交叉编译 | Android 静态库交叉编译 )
文章目录 I . Ubuntu 中交叉编译 Android 动态库 II . Ubuntu 中交叉编译 Android 静态库 I . Ubuntu 中交叉编译 Android 动态库 1 . 要编译 ...
- 【Linux系统编程学习】 动态库的制作与使用
此为牛客网Linux C++课程1.6&1.7 的课程笔记. 1. 动态库命名规则 2. 动态库的制作 第一步,用gcc编译生成.o目标文件,注意要用-fpic参数生成与位置无关的代码: 第二 ...
- 【linux】程序找不到动态库.so的解决办法|查看.so动态库信息|.so动态库加载顺序
目录 找不到.so解决方法 方法一:添加环境变量 方法二:复制so文件到lib路径 方法三:(推荐)添加ldconfig寻找路径 方法四:在编译目标代码时指定该程序的动态库搜索路径 让程序在本目录找到 ...
最新文章
- python与mysql数据库连接中常见错误
- 检测子进程的结束返回状态,status的取值可以是哪些?(简析)
- C语言问题,在位运算中,操作数每右移一位,其结果相当于什么?若左移1位,其结果相当于什么?
- neutron DVR
- Dotfuscator 4.9 破解版下载地址
- pysvn安装及常用方法
- VC程序员常用工具篇
- docx文档怎么排列图片_格式转换太麻烦?仅需一行命令,搞定图片、音频、视频、文本批量转换!...
- 如何在苹果Mac中使用聚焦搜索 NTFS 格式磁盘?
- 卷积神经网络+用pytorch构建神经网络
- maven详解_本地仓库+远程仓库_体现maven用途
- Team Foundation Server 2013 with Update 3 Install LOG
- JAVA图形界面设计(GUI)学生成绩管理系统实战练习之登陆界面
- yigo基础学习笔记5_yigo二次开发配置
- Pytorch 学习(2):神经网络及训练一个分类器(cifar10_tutorial的网络结构图)
- Android中禁止WebView滑动
- jquery.validate.min.js 插件
- 心血来潮,开通了博客
- web自动化时,怎么定位鼠标悬浮时才出现的元素
- 狂神redis学习笔记