【Linux进阶之路】动静态库

文章目录

  • 回顾
  • 一. 静态库
    • 1.代码传递的方式
    • 2.简易制作
    • 3.原理
  • 二. 动态库
    • 1.简易制作
    • 2.基本原理
  • 尾序

回顾

  前面在gcc与g++的使用中,我们简单的介绍了动态库与静态库的各自的优点与区别:

  1. 动态链接库,也就是所有的程序公用一份代码,虽然方便省空间,但是一旦链接库被删,那么所有的程序将无法运行!
  2. 静态链接库,就是所有程序都拷贝一份代码自己用,这样虽然库删除之后会正常运行,但是会使代码的空间异常的大,通常在几十倍到几百倍左右。
  • 详见——基本开发工具

 那么今天就让我们通过动静态库的制作过程与基本原理,进而更深一步了解动静态库吧!

一. 静态库

1.代码传递的方式

  • 我们要想让别人使用我们写的代码,有两种方式:
  1. 将源文件与头文件直接发给别人。
  2. 将源文件打包成的库与头文件发给别人。

区别:

  1. 第一种相当于把实现方法直接发给别人,别人可以进行抄袭与学习以及使用,几乎是把自己的劳动成果(实现方法)拱手让人。
  2. 第二种相当于只把说明书(头文件)发送给了别人,由于打包成了库,因此实现方式别人看不到,只能进行使用。
  • 总结:类比商品,假如你买了个电脑,第一种是只附带有说明书,外加实现的具体方法。第二种是只附带了说明书。因此买的电脑如果是第一种,电脑用坏了,可以自己修,甚至可以自己再造出一台电脑。如果是第二种,用坏了,自己还得去找人家花钱修。
  • 重点:不管哪种方式,头文件必不可少!因为头文件是一份使用说明书,一些具体的使用细节都在头文件中。而且在代码中包含头文件才能用里面的接口。如果不这样使用方式及其麻烦。

2.简易制作

  • 源文件
#include"mymath.h"
int myerrno = 0;
int add(int x,int y)
{return x + y;
}
int sub(int x,int y)
{return x - y;
}
int product(int x,int y)
{return x * y;
}
int div(int x,int y)
{if(y == 0){myerrno = 1;return -1;}return x / y;
}
  • 头文件
//存放的是函数与变量的声明
extern int myerrno;
int add(int x,int y);
int sub(int x,int y);
int product(int x,int y);
int div(int x,int y);

说明:静态库的名称格式为——libXXX.a

生成静态库的指令:

argc -rc [静态库的名称] [要生成静态库的目标文件]
  • Makefile
#定义静态库变量的名称
lib=libmymath.a
#目标文件生成静态库, $(lib)为变量
$(lib):mymath.oar -rc $@ $^
#生成目标文件	
mymath.o:mymath.cgcc -c $^
#清理文件
.PHONY:clean
clean:rm -rf *.a *.o mylib
#将生成的静态库进行打包
.PHONY:output
output:mkdir -p mylib/includemkdir -p mylib/mymathlibcp *.a mylib/mymathlibcp *.h mylib/include
  1. make 生成 静态库与.o文件

在这里插入图片描述
2. make output将静态库与头文件进行拷贝打包

在这里插入图片描述
3. make clean 将多余的文件进行清理

在这里插入图片描述

此时我们的静态库就打包好了,下面我们另起文件进行使用。


与mylib同目录下编写test.c

#include"mymath.h"
#include<stdio.h>
int main()
{printf("myerrno:%d 1 + 0 = %d\n",myerrno,add(1,0));printf("myerrno:%d 1 - 0 = %d\n",myerrno,sub(1,0));printf("myerrno:%d 1 * 0 = %d\n",myerrno,product(1,0));printf("myerrno:%d 1 / 0 = %d\n",myerrno,div(1,0));return 0;
}

我们编译一下:
在这里插入图片描述

  • 可见我们所包含的头文件不在当前目录与默认路径(usr/include),因此找不到。

因此我们需要告诉编译器,去哪找。

gcc test.c -I ./mylib/include

因为头文件已包含文件名,因此不用再说明。

在这里插入图片描述

  • 可见我们函数定义还没有包含,因此找不到定义

因此我们需要告诉编译器,去哪找库。

gcc test.c -I ./mylib/include -L ./mylib/mymathlib/

在这里插入图片描述

  • 可见因为库名字未知且一个目录下可能有多个库,因此我们还找不到定义。

因此我们需要告诉编译器,库名字(库真实的名字为去掉后缀.a 与前缀 lib)。

 gcc test.c -I ./mylib/include -L ./mylib/mymathlib/ -l mymath

在这里插入图片描述

  • 总结
  1. -I(大写 i) 指定头文件的路径
  2. -L指定库所在路径
  3. -l(小写 L) 指定库的名称。且库的名称是去掉 lib 与 .a后缀。

3.原理

  1. 时间:在预处理,编译,反汇编,生成.o(可重定向目标二进制文件)之后。
  2. 动作:将静态库里面的内容,拷贝, 与.o文件一起链接生成的.exe文件。
  • 说明:链接进行段表的合并,符号表的重新定位,其中段表的合并是把有效信息筛选无效信息删除,符号表的重新定位指的时检查代码是否正确,比如函数与某些全局变量的地址是否是有效的。

二. 动态库

1.简易制作

我们还是用之前的代码(将myerrno删了)。

两个关键动作:

  1. 生成.o文件并生成位置无关码
	gcc -FPIC -c mymath.o
  1. 生成动态库
	gcc -shared -o libmymath.so mymath.o
  • Makefile
lib=libmymath.so$(lib):mymath.ogcc -shared -o $@ $^mymath.o:mymath.cgcc -FPIC -c $^ .PHONY:cleanclean:rm -rf *.a *.so *.o .PHONY:outputoutput:mkdir -p lib/includemkdir -p lib/mymathlibcp *.so lib/mymathlibcp *.h lib/include
  1. make生成动态库与.o文件
    在这里插入图片描述

  2. make output 动态库与.h文件进行打包
    在这里插入图片描述

  3. make clean 删除冗余的动态库文件与.o文件
    在这里插入图片描述


同理,我们使用一下库,验证一下。

  • test.c
#include"mymath.h"
#include<stdio.h>int main()
{printf("1 + 1 == %d\n",add(1,1));printf("1 - 1 == %d\n",sub(1,1));printf("1 * 1 == %d\n",product(1,1));printf("1 / 1 == %d\n",div(1,1));return 0;
}

同理我们直接使用之前静态库的结论进行编译链接。

gcc -o test test.c -I lib/include/ -L /lib/mymathlib/ -l mymath

在这里插入图片描述

补充: ldd 【可执行文件】 #显示与可执行文件链接的
  • 可见在生成可执行程序是没问题的,但是显示无法打开这个共享文件对象,这是问什么呢?

解释:

  1. 在动态链接时,我们是在可执行程序变成进程运行的同时,链接到对应库当中,其中库是文件,需要打开才能被链接。
  2. 因此需要让加载器去指定的路径下打开文件,才能使用动态库。
  • 注意:前面的gcc 只是让编译器解决了如何找的问题,如何让加载器打开还没有解决。其次静态链接因为是直接拷贝,因此无需关心打开的问题。

因此:我们需要将让编译器想办法在默认路径下打开库文件。


  1. 直接拷贝到默认路径(最常用)
    在这里插入图片描述
  • 可见是链接成功的,可执行程序也能正常的执行,不过因为要拷贝到系统的路径下,所以我们需要sudo 进行提权。
  1. 在默认路径下建立对应静态库的软链接
    在这里插入图片描述
  • 与第一种方式同理,唯一需要说明的是对不在同一目录下建立软链接,需要使用绝对路径,而不是相对路径。
  1. 修改环境变量LD_LIBRARY_PATH(可能会没有)
    在这里插入图片描述
  • 说明:只需要后跟:与动态库所在的路径即可。至于名称我们在链接形成可执行程序时,已经知道了。
  • 注意:这里环境变量在重启时,就没有了,这是比较恶心的一点。
  1. 添加配置文件
  1. su / su - 切换到root用户
  2. 进入 /etc/ld.so.conf.d/
  3. 添加一个.conf结尾的任意名称的文件
  4. vim 此文件,切换到 Insert模式,添加动态库的路径,保存并退出。
  5. 使用 ldconfig更新此配置文件。
  • 图解:在这里插入图片描述
  • 验证:在这里插入图片描述

2.基本原理

 先来铺垫一下,我们编译器与链接器处理代码的过程:

  1. 预处理,完成头文件的替换,条件编译中代码的裁剪,宏的替换等。
  2. 预编译,完成对语义分析,词法分析,语法分析,符号汇总等,检查语法错误,最终转换为汇编代码。
  3. 汇编,完成符号表与段表的生成,并将代码转换为二进制代码。
  4. 链接,完成符号表的重定位,与段表的合并,并生成可执行程序。

那可执行程序里面存放的是什么呢?

我们反汇编一下:

objdump -S [可执行程序]

在这里插入图片描述

  • 可见是一些指令级别的东西,这里我们或许还能勉强看懂一些汇编,里面还存放着地址。
  • 因此我们可以从中得知,可执行程序在还没有被加载时就已经存在地址了。

那么问题来了,这里的地址是物理地址还是虚拟地址?

  • 肯定是虚拟地址,是要给进程地址空间使用的,物理地址是操作系统在程序加载之后申请的。

这是编译的结论,接下来我们的程序是如何加载到内存当中的呢?

 我们先就可执行程序来进行讨论,我们编译好的可执行程序是在磁盘当中的,在加载时必然要被加载到内存当中。

 对于操作系统来说可执行程序在加载时必然要变为进程,之前我们已经了解过进程是 PCB数据结构, 以及代码和数据。 其中PCB在Linux中为stuct tasks_struct 包含着 页表, 进程地址空间(struct mm_struct) 管理文件的(struct files_struct)等对象。

 那在程序加载时,必然要先形成进程,代码和数据可以后面用时再加载。那进程的地址空间首先要先加载,才能保证后续的正常运行。

至于进程的地址空间的加载,我们用图辅助理解:
在这里插入图片描述

  • 代码在进行加载时,通过页表其指令在进程地址空间中是虚拟地址,也就是编译生成的地址,而实际执行指令的物理地址在加载时就通过页表进行填充。这样进程便可通过指令的虚拟地址通过页表获取到指令的物理地址,进而执行指令。
  • 除此之外,加载时,要想找到可执行程序,还得进程的exe,即可执行程序的路径。这种信息在进程加载时即可进行获取。

 代码现在成功加载到内存中了,那指令是如何运行的呢?

首先万事开头难,如何读取到程序的第一行指令很关键,因此会设置程序入口地址以便接下来的执行

其次CPU首先通过指令寄存器拿到指令的虚拟地址,然后通过页表进行映射,成物理地址,然后根据指令的具体信息,进行执行,然后接着执行下一句代码,如此循环往复。

  • 说明:在加载中,程序指令原本的虚拟地址在内存中变为了物理地址,而原来的虚拟地址则给了进程地址空间,这样才讲的通。

其次数据我们可以在需要时加载,在加载时,触发缺页中断,让操作系统将页表进行填充即可。


代码与数据如何加载我们已经讲的差不多了,那动态库是如何加载的呢?

  • 在这之前我们已经达成了共识,动态库是共享库,即多个进程都可以使用。

那么便可大致画出:
在这里插入图片描述

  • 可见动态库是在加载过程中与进程产生链接的。

那链接到进程地址空间的什么位置呢?

  • 进程地址空间的共享区,这个共享区很大,足够跟多个动态库进行链接。

既然在可能有多个共享库进行链接,那么如何进行链接,才能保证能找到指定的共享库呢?

  1. 我们可以采用起始地址 + 偏移量的方式,从而使函数在在找库时,只需知道偏移量即可。
  2. 偏移量的设定与动态库生成中的位置无关码有关。
  • 拓展:在进行链接时,动态库也可能会产生缺页中断的现象,即用时再进行加载。

补充:

  1. 第三方库,即自己写的库,在进行链接时,必须要指定库名字!
  2. 如果一个库的方法实现有动态库,也有静态库,那么默认优先加载动态库。

  • 总结
  1. 静态库的原理与简易制作。
  2. 动态库的原理与简易制作。
  3. 动态库加载的原理,进程地址空间程序的加载。

尾序

 如果有所帮助的话,不妨点个赞鼓励一下吧!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/159393.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Elasticsearch:FMA 风格的向量相似度计算

作者&#xff1a;Chris Hegarty 在 Lucene 9.7.0 中&#xff0c;我们添加了利用 SIMD 指令执行向量相似性计算的数据并行化的支持。 现在&#xff0c;我们通过使用融合乘加 (Fused Mulitply-Add - FMA) 进一步推动这一点。 什么是 FMA 乘法和加法是一种常见的运算&#xff0c;…

作品Demo:大全

实用小demo 1.3D-地图工程 获取连接&#xff1a;nullhttps://m.tb.cn/h.5kzPTgJ?tkRPBlWep4ZIf 2.3D-饼图 获取连接&#xff1a;https://m.tb.cn/h.5Pt4A8k?tkXRzWWepUnu4https://m.tb.cn/h.5Pt4A8k?tkXRzWWepUnu4

一个月B站涨粉200万,品牌号不可错过的吸粉秘籍

越来越多品牌为了持续在B站营销而创建品牌官方账号&#xff0c;发布原创作品融入B站UP主中&#xff0c;吸引B站用户塑造品牌形象&#xff0c;提高品牌传播度、品牌声量。 据飞瓜数据&#xff08;B站版&#xff09;统计&#xff0c;B站有着超过2万个品牌号&#xff0c;本篇文章…

亚马逊买家号用邮箱怎么注册

想要用邮箱注册亚马逊买家号&#xff0c;那么准备好能接受验证码的邮箱后打开相应的亚马逊官网即可。打开官网后点击注册——输入昵称——输入邮箱——输入密码——接受邮箱验证码并输入&#xff0c;如果遇到需要手机号验证就输入手机号&#xff0c;如果不需要验证&#xff0c;…

7.Gin 路由详解 - 路由分组 - 路由文件抽离

7.Gin 路由详解 - 路由分组 - 路由文件抽离 前言 在前面的示例中&#xff0c;我们直接将路由的定义全部写在 main.go 文件中&#xff0c;如果后面 路由越来越多&#xff0c;那将会越来越不好管理。 所以&#xff0c;下一步我们应该考虑将路由进行分组管理&#xff0c;并且将其抽…

剧情继续:马斯克曝出OpenAI前员工举报信,董事会与奥特曼谈判回归

丰色 发自 凹非寺 量子位 | 公众号QbitAI 经过4天的极限拉扯、反转再反转&#xff0c;奥特曼有可能重新回归了。 据知情人士透露&#xff0c;OpenAI董事会正与奥特曼进行一场“富有成效”的新谈判。 如果奥特曼回到OpenAI&#xff0c;他将继续担任CEO。 与此同时&#xff0c…

【开源】基于JAVA的音乐偏好度推荐系统

项目编号&#xff1a; S 012 &#xff0c;文末获取源码。 \color{red}{项目编号&#xff1a;S012&#xff0c;文末获取源码。} 项目编号&#xff1a;S012&#xff0c;文末获取源码。 目录 一、摘要1.1 项目介绍1.2 项目录屏 二、系统设计2.1 功能模块设计2.1.1 音乐档案模块2.1…

Unsupervised Condition GAN

Unsupervised Condition GAN主要有两种做法&#xff1a; Direct Transformation 直接输入domain X图片&#xff0c;经过Generator后生成对应的domain Y的图像。这种转化input和output不能够差太多。通常只能实现较小的转化&#xff0c;比如改变颜色等。 Projection to Commo…

【C++】:STL中的string类的增删查改的底层模拟实现

本篇博客仅仅实现存储字符(串)的string 同时由于Cstring库设计的不合理&#xff0c;我仅实现一些最常见的增删查改接口 接下来给出的接口都是基于以下框架&#xff1a; private:char* _str;//思考如何不用constsize_t _size;size_t _capacity;//这样写可以const static size_t…

【Windows 常用工具系列 13 -- Confluence 如何快速输入代码块 code block】

文章目录 Confluence 如何快速输入代码块方法二 Confluence 如何快速输入代码块 在使用使用 confluence 进行文档编辑时&#xff0c;有时需要贴上部分代码&#xff0c;但是直接贴代码在 confluence上&#xff0c;显示效果不是太好看&#xff0c;所以confluence 给我们提供了符…

SpringBoot启动后自动打开浏览器访问项目

更简单的一个方法 Runtime.getRuntime().exec("rundll32 url.dll,FileProtocolHandler " url); Springboot项目启动后自动打开浏览器访问(超实用)_浏览器访问springboot项目-CSDN博客 Springboot项目启动后自动打开浏览器访问 1、在Springboot项目中每次启动完项…

PostgreSQL (Hologres) 日期生成

PostgreSQL 生成指定日期下一个月的日期 &#xff08;在Hologres中&#xff0c;不支持递归查询&#xff09; SELECTto_char(T, YYYYMMDD)::int4 AS date_int,date(T) AS date_str,date_part(year, T)::int4 AS year_int,date_part(month, T)::int4 AS month_int,date_part(da…

1.8w 字详解 SQL 优化

来源&#xff1a;捡田螺的小男孩 1、MySQL的基本架构 2、SQL优化 3、explain执行计划常用关键字详解 很多朋友在做数据分析时&#xff0c;分析两分钟&#xff0c;跑数两小时&#xff1f; 在使用SQL过程中不仅要关注数据结果&#xff0c;同样要注意SQL语句的执行效率。 本文…

国内怎么投资黄金,炒黄金有哪些好方法?

随着我国综合实力的不断强大&#xff0c;投资市场的发展也日臻完善&#xff0c;现已成为了国际黄金市场的重要组成部分&#xff0c;人们想要精准判断金市走向&#xff0c;就离不开对我国经济等信息的仔细分析。而想要有效提升盈利概率&#xff0c;人们还需要掌握国内黄金投资的…

用css实现原生form中radio单选框和input输入框的hover样式以及聚焦focus的样式

一.问题描述&#xff1a;用css实现原生form中radio单选框和input的hover已经focus的样式 在实际的开发中&#xff0c;一般公司ui都会给效果图&#xff0c;比如单选按钮radio样式&#xff0c;input输入框hover的时候样式&#xff0c;以及focus的时候样式&#xff0c;等等&#…

Ubuntu安装PCAN-View

目录 一. Hardware 二. Software 2.1 安装驱动 2.2 安装PCAN-View QA 本文介绍如何安装linux版的PCAN-View。 PCAN-View&#xff1a;用来抓包分析CAN/CANFD报文。Hardware: PEAK-System Linux generic #37~22.04.1-Ubuntu SMP PREEMPT_DYNAMIC Mon Oct 9 15:34:04 UTC 2…

Java 最简单的实现 AES 加密和解密

AES简介 AES&#xff08;Advanced Encryption Standard&#xff09;高级加密标准&#xff0c;是一种被广泛使用的对称加密算法&#xff0c;用于加密和解密数据。它曾经是美国政府的一个机密标准&#xff0c;但现在已成为公开的加密算法&#xff0c;并被广泛使用于商业、政府及…

第四代智能井盖传感器:智能井盖位移监测

当城市道路上的井盖出现异常时&#xff0c;可能会导致突发的交通事故或人员受伤事件。而传统的井盖监测往往依靠人力进行巡查&#xff0c;这种方式可能会因为监测不及时或不准确而带来问题。但是现在有了智能井盖传感器&#xff0c;它们成为了城市地下生命线的守护者。这种智能…

【开源】基于Vue.js的开放实验室管理系统的设计和实现

项目编号&#xff1a; S 013 &#xff0c;文末获取源码。 \color{red}{项目编号&#xff1a;S013&#xff0c;文末获取源码。} 项目编号&#xff1a;S013&#xff0c;文末获取源码。 目录 一、摘要1.1 项目介绍1.2 项目录屏 二、研究内容2.1 实验室类型模块2.2 实验室模块2.3 实…