理解静态库、动态库加载

个人主页:Lei宝啊 

愿所有美好如期而遇


系统角度理解

我们先来谈谈进程地址空间,当我们将一个可执行程序跑起来的时候,操作系统首先会在内存中创建出task_struct,也就是进程控制块,然后将可执行程序的代码和数据加载进内存,和进程地址空间的地址通过页表建立映射关系。

我们提到了进程虚拟地址空间,但是没有提到过他的数据是怎么来的,这个我们后面会说,现在这里想说的是,可执行程序链接静态库后,运行时不再需要静态库,因为静态库的代码已经拷贝了一份到可执行程序中,但是动态库不是这样,也就是说,当可执行程序执行到了某行代码,这行代码调用的是动态库的内容,那么我们其实是需要动态库的。

所以也就是说,即使动态库文件不是和程序一起加载进内存,那么在程序需要时,仍然需要加载进内存,并且动态库将被映射到进程地址空间的共享区,所以动态库也叫做共享库。

如果我们有多个进程同时需要这一个库,那么现在我们假设,有一个进程调用这个库,如果这个库没有被加载进内存,那么就先将他加载进来,然后映射到进程的共享区,如果说已经加载进来了,那么就直接映射。

本质也就是进程中公共的代码和数据,只需要一份,这样就提高了效率,所以一般经常被使用的文件我们打包成动态库,而不是静态库。

编址问题

首先我们提出可执行程序是有自己的格式信息的,并且在被加载进内存时就已经被划分出了各个区域:

text就是代码区域,data就是已经初始化的数据区,bss就是全局未被初始化的数据。 

这里我们抛出一个问题:可执行程序中存在地址吗?

是的,存在,而且进程地址空间中的数据是使用程序中的数据进行初始化的,但是是怎么初始化的呢?进程地址空间是一个结构体,记录着每一个区域的起始和结束,也就是说,既然你程序可以初始化这样的区域,而且你程序中存在地址,也就是说,程序必然有自己的一套编址方式喽,这里我们又要说到,进程地址空间是由操作系统维护,而程序是编译器编译的,也就是说,操作系统和编译器也是有关系的吗?也就是说,虚拟地址空间不仅操作系统要维护,编译器在编址时也要遵守这套规则。

这里我们提出两个概念,绝对编址和相对编址,绝对编址就像是一个坐标系的原点,我们所写的坐标都是按照原点来确定,相对编址就像是给定一个坐标点,其他坐标相对于这个坐标的位置。

我们现代的编译器是按照这种绝对编址,也就是平坦模式进行编址的,过去的编译器是按照相对编址,给出一个段地址,下面的代码地址都是相对于段地址的偏移量,而我们现在的绝对编址其实也可以按照这种方式来理解,只不过我们的段地址只有一个,就是0x0000 0000,剩下的偏移量其实也就是地址,也就是说,我们可以认为我们的可执行程序只有一个区域。

现在,我们可以说到,虚拟地址其实就是这些地址,在程序加载进内存后,使用这些地址初始化进程地址空间,然后用实际的物理地址在页表中构建映射关系。

我们可以通过反汇编来看一下程序中的地址:

使用objdump -S 可执行程序 > 一个文件中

如果是相对编址的话,那么<_init>下面那些地址应该是000000,000004这样的相对地址。 

而这些地址,不正是我们将来的虚拟地址吗?!我们在调试时看到的地址也就是这些地址!

理解动态库链接和加载问题

一般程序的加载(链接静态库):

假设我们有一个test.c文件,当我们将他编译成,o文件时,这个文件里的代码就已经有了地址,有了他自己的格式,并且是按照相对编址方式进行编址,静态库也是如此。

当我们将这个.o文件和静态库链接形成可执行程序时,将会将他们的地址以绝对编址的方式重新进行编址,可执行程序会将静态库中需要的代码拷贝进来,最终就是我们上面可执行程序的样子。

当我们将这个可执行程序跑起来时,首先操作系统会先在内存中创建task_struct,此时页表和进程地址空间也会创建,接着将可执行程序的代码和数据加载进内存,此时可执行程序内部有逻辑地址,加载进内存我们称之为虚拟地址,并且现在这些代码有了他们的物理地址,此时就会将这些虚拟地址和物理地址在页表中建立映射关系,并且用可执行程序表头中的数据初始化进程地址空间(mm_struct),并且将pc指针初始化为main的虚拟地址。

接下来,就要开始执行正文部分的代码了,有了pc指针,将他存的虚拟地址通过MMU和页表转化为物理地址,找到物理地址中存储的指令,将其加载进指令寄存器中,就可以开始执行这行指令,执行过这句指令后,pc指针会找到下一条指令的虚拟地址,并继续通过MMU和页表转化为物理地址......,这样,我们的程序就执行起来了。

动态库的加载:

动态库里代码的地址也是根据相对编址进行的编址,同时,可执行程序链接动态库时,不会将代码拷贝进去,只会留下库名+方法的偏移量。

当我们将程序和动态库加载进内存,即将执行到target+0x01这行指令时,程序计数器中保存着他的虚拟地址,根据他的虚拟地址找到他的物理地址,将这行指令加载进指令寄存器中,解析时,根据这个库名找到库的起始虚拟地址,然后这个虚拟地址加上偏移量找到方法的虚拟地址,在MMU和页表转化为物理地址后找到方法的物理地址。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

整数和浮点数在内存中存储

整数在内存中的存储 整数的2进制表⽰⽅法有三种&#xff0c;即原码、反码和补码。 对于整形来说&#xff0c;数据存放内存中的其实是补码。 在计算机系统中&#xff0c;数值一律用补码来表示和存储。原因是&#xff0c;使用补码&#xff0c;可以使符号位和数值域统一处理&am…

ARMday7

VID_20240322_203313 1.思维导图 2.main.c #include"key_inc.h" //封装延时函数 void delay(int ms) {int i,j;for(i0;i<ms;i){for(j0;j<2000;j){}} } int main() {//按键中断的初始化key1_it_config();key2_it_config();key3_it_config();while(1){printf(&q…

备战蓝桥杯Day34 - 每日一题

题目描述 解题思路 1.输入数据n&#xff0c;并将字符串类型转换成整数类型 2.求出输入n是2的几次幂&#xff08;调用math库中的求对数的方法&#xff09;&#xff0c;在下面的循环中要用到 3.定义sum和&#xff0c;将抽取到的牌的总数加起来存储 4.count 0 # 记录 2 的第几…

台达变频通过Modbus转Profinet网关可以在环网冗余中使用

Modbus转Profinet网关&#xff08;如XD-MDPN100&#xff09;是一种能够实现Modbus协议与Profinet协议之间转换的设备。它支持Modbus RTU协议和Profinet协议还支持MRP环网冗余系统&#xff0c;,可以通过配置软件进行协议转换&#xff0c;使得原本只能使用Modbus协议的设备可以与…

微服务day05(中) -- ES索引库操作

索引库就类似数据库表&#xff0c;mapping映射就类似表的结构。 我们要向es中存储数据&#xff0c;必须先创建“库”和“表”。 2.1.mapping映射属性 mapping是对索引库中文档的约束&#xff0c;常见的mapping属性包括&#xff1a; type&#xff1a;字段数据类型&#xff0c;…

中兴通讯服务器荣获滴滴“最佳需求响应「和衷共济」奖”

在数字经济加速发展的背景下&#xff0c;算力成为数字产业的核心支撑力量&#xff0c;而服务器和存储产品更是为互联网创新体验提供了底层基础设施保障。在此背景下&#xff0c;中兴通讯服务器产品有效支撑滴滴出行智慧交通解决方案&#xff0c;凭借卓越表现&#xff0c;获得滴…

QGraphicsView的使用,view坐标,scene坐标,item坐标

Graphics View绘图构架 QGraphicsScene&#xff08;场景&#xff09;&#xff1a;可以管理多个图形项QGraphicsItem&#xff08;图形项&#xff09;&#xff1a;也就是图元&#xff0c;支持鼠标事件响应。QGraphicsView&#xff08;视图&#xff09;&#xff1a;关联场景可以让…

基于python+vue学生作业管理系统flask-django-nodejs-php

快速发展的社会中&#xff0c;人们的生活水平都在提高&#xff0c;生活节奏也在逐渐加快。为了节省时间和提高工作效率&#xff0c;越来越多的人选择利用互联网进行线上打理各种事务&#xff0c;然后线上管理系统也就相继涌现。与此同时&#xff0c;人们开始接受方便的生活方式…

IDEA/Android Studio格式化代码快捷键失效的解决

问题描述 用AS写一个项目的时候&#xff0c;发现CtrlAltL的格式化快捷键并不生效&#xff0c;按下之后没有任何反应。而格式化文件快捷键CtrlAltShiftL依旧生效&#xff0c;难道是设置出了问题&#xff1f; 打开设置页面发现快捷键设置很正确&#xff0c;并没问题&#xff0c…

掌握C语言结构体,开启编程新世界

✨✨欢迎&#x1f44d;&#x1f44d;点赞☕️☕️收藏✍✍评论 个人主页&#xff1a;秋邱博客 所属栏目&#xff1a;C语言 &#xff08;感谢您的光临&#xff0c;您的光临蓬荜生辉&#xff09; 前言 前面我们也涉及到了结构体的讲解&#xff0c;但是只是粗略的讲了一下。 接下…

力扣hot100题解(python版91-95题)

91、不同路径 一个机器人位于一个 m x n 网格的左上角 &#xff08;起始点在下图中标记为 “Start” &#xff09;。 机器人每次只能向下或者向右移动一步。机器人试图达到网格的右下角&#xff08;在下图中标记为 “Finish” &#xff09;。 问总共有多少条不同的路径&…

java分割等和子集(力扣Leetcode416)

分割等和子集 力扣原题链接 给你一个只包含正整数的非空数组nums。请你判断是否可以将这个数组分割成两个子集&#xff0c;使得两个子集的元素和相等。 示例 1&#xff1a; 输入&#xff1a;nums [1,5,11,5] 输出&#xff1a;true 解释&#xff1a;数组可以分割成 [1, 5, 5] …

【深度学习基础知识】IOU、GIOU、DIOU、CIOU

这里简单记录下IOU及其衍生公式。   为了拉通IOU及其衍生版的公式对比&#xff0c;以及方便记忆&#xff0c;这里用一个统一的图示来表示出所有的参数 【&#xff21;】目标框的区域【&#xff22;】预测框的区域【&#xff23;】&#xff21;与&#xff22;的交集【&#xff…

关于TEMU 亚马逊美国哺乳枕(Nursing Pillow)法规16 CFR 1242介绍

美国首个哺乳枕法规16 CFR 1242发布 美国消费品安全委员会CPSC于2023年9月26日在联邦公报上发布了哺乳枕新法规16 CFR 1242的草案&#xff0c;旨在降低哺乳枕使用带来的伤害和死亡风险。该法规草案提到了在使用哺乳枕时由于婴儿睡着或无人看护而导致的窒息、陷落和跌落风险。在…

2024:RAG年

如果 2023 年都是关于 ChatGPT 和 Llama-2 等基础LLM&#xff0c;那么我的预测是 2024 年将是关于检索增强一代&#xff08;RAG&#xff09;的。 在这篇博文中&#xff0c;我阐述了为什么 RAG 将在 2024 年飞速发展&#xff0c;不仅是企业采用率&#xff0c;而且消费者采用率也…

BigDecimal类的使用,用于精确计算任意精度的数字

BigDecimal类 BigDecimal 是 Java 中用于精确表示任意精度的十进制数的类。在很多情况下,使用基本数据类型(如 double 或 float)进行浮点数计算可能会导致精度丢失或舍入错误。BigDecimal 提供了一种更精确的解决方案,可以处理需要高精度计算的场景,比如财务应用或科学计算…

记录解决问题--activiti8.2 流程图图片由png改为svg前端不显示图片问题

1.说明 如果是vue svg显示&#xff0c;请查阅其他标准资料&#xff0c;类似使用svg标签。我这里讲的另外一种情况&#xff0c;链接返回的是svg文件&#xff0c;需要用v-html显示图片。 2.activiti6流程图图片格式 ①png格式。可以查看链接返回&#xff0c;以png开头。 ②前端…

蓝桥杯练习——神秘咒语——axios

目标 完善 index.js 中的 TODO 部分&#xff0c;通过新增或者修改代码&#xff0c;完成以下目标&#xff1a; 点击钥匙 1 和钥匙 2 按钮时会通过 axios 发送请求&#xff0c;在发送请求时需要在请求头中添加 Authorization 字段携带 token&#xff0c;token 的值为 2b58f9a8-…

适合新生儿的奶瓶有哪些?五款高分新生儿奶瓶分享!

每一个有新生儿的家庭都一定会挑选奶瓶&#xff0c;但是因为市面有太多品牌和款式&#xff0c;让大家难以挑选&#xff0c;更为重要的是还有可能会不小心选到劣质的产品&#xff0c;不仅奶嘴的仿真度差、易胀气&#xff0c;还可能高温消毒后散发有害物质&#xff01;那么新生儿…

力扣 字符串解码

维护一个放数字的栈&#xff0c;一个放字母的栈 遇到[把数字和字母入栈&#xff0c;遇到]把当前字母循环加上数字栈头遍的字母栈头 class Solution { public:string decodeString(string s) {string ans"";stack<int>sz;stack<string>zm;里面是string …