进程——环境变量及程序地址空间

目录

环境变量

概念

补充:命令行参数

引入

其它环境变量

理解

程序地址空间

引入

理解

虚拟地址存在意义


环境变量

概念

环境变量一般是指在操作系统中用来指定操作系统运行环境的一些参数。打个比方,就像你布置房间,这些参数就类似房间里的布置规则,决定了东西怎么摆放、怎么用,能让操作系统在合适的"环境"下正常运转 。

补充:命令行参数

命令行参数是在运行命令行程序时,跟随在程序名称之后输入的额外参数。这些参数用于控制程序的行为等。大家可能认为main是没有参数,实则不然实际上它是有参数的。

上面的argv表示命令行参数的数组,argc表示数组里内容的个数。下面是它的运行结果:

其实通过上面的我们可以发现我们在命令行输入的指令,其实是字符串。

在执行该可执行文件时,bash会创建一个子进程用来执行该程序。命令行参数会被bash解析和切分,然后bash会通过内部的机制来处理和传递这些参数信息给要执行的程序,由程序自身来存储和处理命令行参数。

上面是运行结果是不是感觉很熟悉,其实我们平时使用的指令,它里的选项就是这样实现的,通过上面就可以实现在一个函数里有不同的子功能。

引入

不知道大家有没有想过,为什么在执行系统程序时不同写路径,而我们自己写的程序就要写路径?

因为存在环境变量------PATH,它的作用是告诉系统去哪里查找。env:查看所有环境变量,echo $环境变量名:查看指定环境变量。环境变量默认是放在~/.bash_profile里。

我们可以添加路径:PATH=$PATH:新路径。不要PATH=新路径否则会替换原有路径!

其大概过程是,如果没有指定路径bash会先查看是不是内建命令,如果不是它就会去PATH里所有路径下找,如果找到了就执行该文件,没找到就会显示错误信息。补充:内建命令是在bash里面的,它是bash自己解释处理的,不需要创建子进程。

环境变量一般是从配置文件里来的,每次从新登陆就会从配置文件里读取一次。

其它环境变量

下面我介绍几个常用的环境变量:

  • HOME:当前用户所对应的家目录(~),这就解释了为什么cd ~就会回到家目录。
  • SHELL:shell所在路径,一般指的是bash所在路径
  • USER:当前用户是谁,以便进行权限管理、个性化设置等操作
  • LOGNAME:登录是谁,他与上面的USER在大多数情况下是一样的
  • HISTSIZE:bash会记录HISTSIZE个历史命令
  • HOSTNAME:登录主机名

这里我不知道为什么我没有。

  • PWD:当前工作目录
  • OLDPWD:上次工作目录(-)

这里我不知道为什么我没有。

理解

其实通过getenv(name)也可以具体查看某环境变量,还有environ它是访问环境变量的全局变量。

在bash里有一张环境变量表,它是可以被子进程继承的,并且子进程的修改不影响父进程。所以环境变量具有全局性和独立性。在main函数里还有一个参数它是用来存储环境变量表(main函数里最多三个参数)。

通过export可以自己导入环境变量,unset取消自己导入的环境变量。

我上面所说的都是bash里的环境变量。但大家有没有想过export它是一个子进程啊,我用export导了一个环境变量那不是应该在子进程里吗?怎么会到bash里呢?其实export它是内建命令,它是bash亲自执行的。

命令行是可以定义变量的(命令行定义),这种变量被称为本地变量(支持该操作有一点原因是要支持脚本),它不会被继承,只能在当前的进程里使用。用set可以显示环境变量和本地变量,用unset也可以取消本地变量。

注意不要有空格,否则就是命令了。本地变量是可以变成环境变量的。

程序地址空间

引入

大家先看下面的代码:

是不是感觉不可思议,一块地址上竟然有两个值!实际上我们看到的地址是虚拟地址并不是物理地址,真正的物理地址用户是看不到的。

理解

一个进程有一个虚拟地址空间,一套页表。页表是虚拟地址到物理地址映射。所以上面的代码就可以这么理解。

首先是一个父进程

(上面写错了是命令++++参数环境变量)然后fork一个子进程,子进程会复制父进程的PCB和页表,并自己做出一些修改

当子进程对val修改后

虚拟地址空间的大小在32位机器上是4G,一般是用户用3G,内核用1G。进程会以为自己占据了整个内存。它里面的区域划分包括如代码段、数据段、堆、栈等。系统里的每个进程都会有自己的进程地址空间。

补充:可以简单认为进程地址空间和虚拟地址空间是一回事。但严格来说,虚拟地址空间是机制,进程地址空间是实例。也就是说进程地址空间是虚拟地址空间在单个进程中的具体实现。

操作系统会对进程地址空间进行管理,那怎么管理呢?依然是先描述在组织,所以PCB里有mm_struct,它是对进程地址空间的描述。

mm_struct是Linux内核中用于描述进程内存空间的数据结构。它包含了进程虚拟地址空间的各种信息。它里面主要是记录区域的开始位置和结束位置。当想调整某区域时,只需要改变起始位置和结束位置即可。

堆是不连续的,我们需要记录每一个堆的开始和结束位置。但是在mm_struct里只有一对用于记录堆的开始和结束位置,那如果我又用malloc开辟了一段空间,那该怎么办呢?其实在mm_struct里有vm_area_struct它表示一个独立的虚拟内存区域。所以不管开辟多少空间只需要用vm_area_struct连接起来即可。其实不仅是堆,其它的区域也会用vm_area_struct表示区域范围。vm_area_struct通过双链表和红黑树管理起来。

mm_struct是对进程整个虚拟地址空间进行宏观的,整体的描述和管理。 vm_area_struct是更细致地表示虚拟地址空间中的每一个具体的、独立的虚拟内存区域,它描述了每个区域的具体属性。

int有4个字节,它占据了4个虚拟地址。在取它地址时为什么只返回一个地址?其实系统会返回地址最小的那个值,剩下的可以通过偏移量来计算出来,所以才会有类型这一说。

虚拟地址存在意义

  • 程序的各个部分(如代码段、数据段等)在物理内存上的存储位置可能是不连续、无序的,但我们通过虚拟地址来看就是有序的。
  • 在地址的转化过程中,会对你的地址和操作进行合法性判断,从而保护物理内存。

比如:

当你拿野指针去查页表,页表里面并没有该地址,程序就会崩溃(也不是说一定会崩溃,可能你的野指针正好在页表里,这其实比崩溃还可怕)。

当执行char* str = "hello"; *str = 'H'; 时,查页表时会发现str只有只读权限没有修改权限,程序就会崩溃。

  • 让进程管理和内存管理进行一定程度的解耦和。

有了虚拟地址之后我们就不用把所有的代码和数据给一次性都给加载到物理内存里,而是分批次加载,甚至可以不加载,让进程只有PCB和页表,这时去查页表时如果物理地址为空就会发生缺页中断,进程会停下来等页表补充完后再次执行。

相关推荐
Carlos_sam19 分钟前
OpenLayers:封装Tooltip
前端·javascript
工呈士33 分钟前
MobX与响应式编程实践
前端·react.js·面试
嘉小华34 分钟前
Android Lifecycle 使用
前端
Sherry00736 分钟前
实时数据传输协议:WebSocket vs MQTT
前端·websocket
然我36 分钟前
JavaScript的OOP独特之道:从原型继承到class语法
前端·javascript·html
腹黑天蝎座38 分钟前
如何更好的实现业务中图片批量上传需求
前端
嘉小华39 分钟前
Android Lifecycle 源码解析
前端
不_喜40 分钟前
游戏开发零散知识点和优化记录
前端
去伪存真1 小时前
提交规范靠吼没用,看我用“shell+husky螺丝刀”,一键给40多个项目上锁
前端·eslint
翠莲1 小时前
vue3+TS+eslint9配置
前端·代码规范