Linux——进程基本概念下篇

Linux——进程基本概念下篇


文章目录

  • Linux——进程基本概念下篇
  • 一、环境变量
    • 1.1 环境变量的定义
    • 1.2 环境变量的相关命令
    • 1.3 命令行参数
    • 1.4 本地变量和环境变量
    • 1.5 常规命令和内建命令
  • 二、进程地址空间
    • 2.1 地址空间的概念
    • 2.2 页表和MMU
    • 2.3 地址空间的作用
    • 2.4 地址空间的好处


一、环境变量

要执行一个命令,必须先要找到对应的可执行程序
为什么我们运行自己的程序的时候要加上././就是指明当前路径,要找到程序才能运行,而我们运行指令的时候却不用加呢?

因为环境变量

我们通过Xshell输入账号密码登录云服务器
在这里插入图片描述
在这里插入图片描述
为什么命令行中会自动进入我所在的家目录/home/hs 而root进入的是/root

登陆的时候,系统会进行以下步骤
1.输入用户名&&密码
2.认证
3.形成环境变量(肯定不止一个,PATH,PWD,HOME) 并且根据用户名,初始化HOME=/root,HOME=/home/XXX
4.cd $HOME

1.1 环境变量的定义

环境变量一般是指在操作系统中用来指定操作系统运行环境的一些参数

环境变量通常具有某些特殊用途,还有在系统当中通常具有全局特性(这就是我们使用某些程序不需要./的原因)

系统中会存在大量的环境变量,每一个环境变量都有它自己的特殊用途,用来完成特定的系统功能
注意环境变量也是OS在内存或磁盘中开辟的空间用来保存系统相关数据

常见的环境变量

PATH : 指定命令的搜索路径
HOME : 指定用户的主工作目录(即用户登陆到Linux系统中时,默认的目录)
SHELL : 当前Shell,它的值通常是/bin/bash

1.2 环境变量的相关命令

echo $NAME // NAME:你的环境变量名称

在这里插入图片描述

env命令可以查看系统所有变量

在这里插入图片描述
我们也可以把我们自己的程序路径加到环境变量中

使用export指令

export PATH=

注意如果直接export PATH=我们自己的程序路径的话,会导致path环境变量本身具有的环境变量被覆盖
如果之前的path环境变量被覆盖,那么系统中的很多指令就会执行不了,因为没有默认寻找路径

所以不改变原有环境变量,并且添加自己的环境变量推荐以下写法

export PATH=$PATH:xxx程序所在路径 向环境变量PATH中添加xxx程序

默认更改环境变量,只限于本次登陆,重新登陆,环境变量自动被恢复 为什么?

我们直接更改的是bash进程内部的环境变量信息,每一次重新登陆,都会给我们形成新的bash解释器并且新的bash解释器自动从读取形成自己的环境变量表信息

1.3 命令行参数

main函数可以携带两个参数

int main(int argc, char* argv[])

int argc是参数的个数
char* argv[]是一个数组指针,数组的每个元素是char* 类型的指针,在linux中,通常以每个程序后面的附加选项实现,例如ls -a, ls -l,会被系统以空格为分隔符,划分成一个个字符串,依次放入argv[]中

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

shell or os自动帮我们做的,将一个大的字符串以空格作为分隔符,被分割成了若干个字串

为什么要这么干? ? ?

命令行参数,可以支持各种指令级别的命令行选项的设置,理解历史学的指令与选项之间的关系

其实main函数还有第三个参数——如何证明?

在这里插入图片描述
在这里插入图片描述
再执行env指令
在这里插入图片描述
通过对比,我们发现第三个参数是系统中的环境变量,那么这是谁传给main函数的呢?

系统启动我们的程序的时候,可以选择给我们的进程(main)提供两张表:
1.命令行参数表
⒉环境变量表

命令行启动的进程都是shell/bash的子进程,子进程的命令行参数和环境变量是父进程bash给我们传递的

父进程的环境变量信息又从哪里来?

环境变量信息是以脚本配置文件的形式存在的,每一次登陆的时候,你的bash进程都会读取vim.bash_profile配置文件中的内容,为我们bash进程形成一张环境变量表信息

如何证明?

当操作系统运行起来后,bash进程也会运行起来,同时读取配置环境变量的本地脚本,形成环境变量表,而我们的进程又大多都是bash的子进程,都会被传这张相同的环境变量表,系统环境变量具有全局属性,可以被子进程继承下去

即使不在main函数中读取第三个参数,main函数中也会有一个全局变量char** environ指向这张环境变量表
在这里插入图片描述

1.4 本地变量和环境变量

本地变量只在bash进程内部有效,不会被子进程继承下去
环境变量通过让所有的子进程继承的方式,实现自身的全局性

shell本身就是一个c程序,本地变量相当于在main中定义的一些局部变量,用set查看shell中的环境变量和本地变量,unset取消环境变量和本地变量

举个例子
在这里插入图片描述

1.5 常规命令和内建命令

Linux的命令分类

常规命令:shell调用fork函数,创建子进程,让子进程成执行的
内建命令:shell命令行的一个函数,可以直接读取shell内部定义的本地变量

echo,set,unset,pwd,umask,export等等都是内建命令,内建命令速度更快,不受环境变量影响

二、进程地址空间

2.1 地址空间的概念

在我们之前学习的C语言或者C++中,相信大家对内存应该有一定的了解,对于内存中的分布,也应该有一定的概念

在我们之前的学习中,内存中的空间分布应该是如下图中所示
在这里插入图片描述
在这里插入图片描述
但实际在内存中真的是这样分布的吗?

我们用代码进行验证
在这里插入图片描述
输出结果
在这里插入图片描述

此时能得到在父进程和子进程中g_val的地址和值是相同的,没有问题!
因为根据我们之前学的,子进程创建的时候,代码和数据是以父进程为模板创建的

我们将代码稍稍修改一下
在这里插入图片描述
输出结果
在这里插入图片描述

我们发现,父子进程,输出地址是一致的,但是变量内容不一样,这就与我们知道的知识相悖了,为什么同一个变量,同一个地址,会出现两个不同的值?

前面我们讲过如果父子进程之间的任何一方将值修改了后,OS就会进行写时拷贝,但如果发生写实拷贝的话,地址为什么还一样呢?

所以我们可以得出,此处的地址绝对不是真实的物理地址,我们平时用到的地址,都是虚拟地址/线性地址

结论

变量内容不一样,所以父子进程输出的变量绝对不是同一个变量
但地址值是一样的,说明该地址绝对不是物理地址
在Linux地址下,这种地址叫做虚拟地址
我们在用C/C++语言所看到的地址,全部都是虚拟地址!物理地址,用户一概看不到,由OS统一管理

OS必须负责将 虚拟地址 转化成 物理地址

什么是地址空间? 什么是区域划分?

OS中有物理地址,而每个进程都有一份自己的虚拟地址
每个进程都会认为自己是独享整个物理地址,它们都会以统一的方式划分自己的内存

每个进程都有自己的地址空间,而操作系统需要管理这些地址空间,如何管理?

答案很简单,先描述,再组织

所以进程地址空间本质是内核的数据类型(struct mm_struct)

struct mm_struct
{
	long code start;
	long code_end;
	long data_start;
	long data_end;
	long heap_start;
	long heap_end; //brk
	long stack_start;
	long stack_end;
	................
	//其他区域也类似
}

在这个结构体中划分好了各个区域的起始和终止位置,这就是区域划分
通过这些区域划分出不同的界限,每个进程都认为自己的mm_struct代表整个内存(地址从0x000000…000 ~ 0xFFFFFF…FFF),也就是每个进程都认为自己拥有4GB的空间

2.2 页表和MMU

OS通过页表+MMU(内置在CPU中的一个硬件)来对齐进行映射
页表负责把虚拟地址映射到物理地址

那么为什么需要页表映射,不能直接让进程地址空间访问物理内存吗?

如果我们允许进程直接访问物理内存,就很有可能访问到不属于自己的空间,所以我们需要页表来管理,如果是非法的OS就直接可以拒绝访问,所以页表中除了虚拟地址和物理地址之间的映射,还有对应的访问权限字段和是否分配&&是否有内容的字段

在这里插入图片描述

在这里插入图片描述

举个例子:

const char* p = "abcd",我们不能通过指针p修改"abcd",因为它在字符常量区,本质是OS给我们的权限只有读权限(通过页表的权限管理)

值得注意的是,当进程创建子进程前,父子进程在页表中的数据段和代码段的权限都会被OS修改成只读的,代码段只读是正常的,因为代码本身就不能被更改,但为什么数据段也要修改成只读呢?
父进程创建子进程的时候首先将自己的读写权限,改成只读,然后再创建子进程

当父进程创建子进程之后,子进程写入,此时发生写时拷贝,重新申请空间,进行拷贝,但由于权限是只读的,OS就会进行介入,此时OS发现是因为需要进行写实拷贝的情况,会进行放行,OS会修改页表,父子进程指向不同的物理内存,这是写实拷贝的策略机制

但当不是这种情况的时候,就比如上述例子中,本身字符常量区是不能修改的,此时程序出错,OS就会将请求驳回

在这里插入图片描述

2.3 地址空间的作用

为什么要有地址空间?

1. 让进程以统一的视角看待内存,所以任意一个进程,可以通过地址空间+页表可以将乱序的内存数据,变成有序,分门别类的规划好
2. 存在虚拟地址空间,可以有效的进行进程访问内存的安全检查
3. 将进程管理和内存管理进行解耦,通过页表让进程映射到不同的物理内存处,从而实现进程的独立性

2.4 地址空间的好处

我们想象一种场景:我们直接申请一块大空间,操作系统会直接全部给我了吗?如果我们没有用那么多,其他的空间不是被浪费了吗?

所以当我们在堆区申请一块空间的时候,OS会把地址空间中的heap_end加上我们申请的大小,但是却没给我们物理内存,当我们需要读取的时候,才会使用页表进行映射,得到物理空间

假设没有地址空间,那么CPU是不是只能遍历一遍物理空间才能找到该进程的起始位置?

我们可以把main函数的地址放入每个进程的地址空间(同一位置),然后CPU就访问地址空间的固定位置然后通过页表映射找到物理内存存放代码数据的位置,这样效率可以得到提升

总结
为什么要有地址空间?

1️⃣ 通过添加一层软件层,完成对进程操作的风险管理(权限管理),保护了物理内存以及各个进程的数据安全
2️⃣ 将内存申请和内存使用的概念在时间层面上划分清楚,通过地址空间完成屏蔽底层申请物理空间的过程,达到进程读写内存和OS进行内存管理操作,在软件层面上分离(你不知OS给你的内存是富裕或者是快满状态的空间)
3️⃣ 站在CPU和应用层的角度,每个进程可以看作统一使用4GB空间,而且每个空间区域的相对位置是比较确定的

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

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

相关文章

Docker容器:docker基础

目录 一.docker前言 云计算服务模式 关于 docker 产品 虚拟化产品有哪些? ① 寄居架构 ② 源生架构 虚拟化技术概述 主流虚拟化产品对比 1. VMware系列 2. KVM/OpenStack 3. Xen 4. 其他半/全虚拟化产品 二. docker 的相关知识 1. docker 的概念 doc…

【古琴】倪诗韵古琴雷修系列(形制挺多的)

雷音系列雷修:“修”字取意善、美好的,更有“使之完美”之意。精品桐木或普通杉木制,栗壳色,纯鹿角霜生漆工艺。 方形龙池凤沼。红木配件,龙池上方有“倪诗韵”亲笔签名,凤沼下方,雁足上方居中位…

码头船只出行及配套货柜码放管理系统-毕设

毕业设计说明书 码头船只出行及配套货柜码放 管理系统 码头船只出行及配套货柜码放管理系统 摘要 伴随着全球化的发展,码头的物流和客运增多,码头业务迎来新的高峰。然而码头业务的增加,导致了人员成本和工作量的增多。为了解决这一基本问题&…

Bentley二次开发教程24-交互式类工具

交互式工具概念简述 本次内容主要涉及到交互式工具的使用,在MicroStation中,超过一半的功能都是以交互式工具的形式而存在的,因此交互式工具在MicroStation二次开发中便显得非常重要。当我们的鼠标或键盘在视图中产生交互操作时,…

各平台奇怪问题备忘录

微信小程序 小程序报错Page 页面路径 has not been register yet 描述:uniapp做微信小程序开发时,新增某页面后,小程序跳转该页面报错Page 页面路径 has not been register yet 已知:page.json已添加该页面,小程序a…

【Linux】文件目录及路径表示

1. Linux目录结构 在 Linux 系统中,有几个目录是比较重要的,平时需要注意不要误删除或者随意更改内部文件。 /etc: 这个是系统中的配置文件,如果更改了该目录下的某个文件可能会导致系统不能启动。 /bin, /sbin, /usr/bin, /usr…

vue快速入门(四十一)组件通信-依赖注入

注释很详细&#xff0c;直接上代码 上一篇 新增内容 祖先组件向下传值子代组件接受数据 源码 App.vue <template><div id"app"><sonComponent></sonComponent></div> </template> <script> import sonComponent from &qu…

python绘制随机地形地图

&#x1f47d;发现宝藏 前些天发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&#xff0c;风趣幽默&#xff0c;忍不住分享一下给大家。【点击进入巨牛的人工智能学习网站】。 当我们谈论计算机编程中的地图生成时&#xff0c;通常会想到游戏开发、仿真模拟或者数据可视…

vue3 修改路由中的meta属性

有些时候可能需要在路由跳转前后修改meta里面的相关属性值&#xff0c;这个时候就需要使用钩子函数&#xff08;路由守卫&#xff09;&#xff0c;钩子函数有全局钩子&#xff0c;局部组件钩子函数以及路由配置里面的钩子函数 &#xff08;这些也叫路由守卫&#xff09; 1.全局…

python数字验证码自动识别

&#x1f47d;发现宝藏 前些天发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&#xff0c;风趣幽默&#xff0c;忍不住分享一下给大家。【点击进入巨牛的人工智能学习网站】。 在网络上&#xff0c;许多网站和应用程序使用验证码&#xff08;Completely Automated Publ…

Ubuntu系统开机长

Ubuntu系统开机长 1. 检查开机自启动软件的所占时间2. 将耗时最高的禁止开机自启动 1. 检查开机自启动软件的所占时间 systemd-analyze blame2. 将耗时最高的禁止开机自启动 sudo systemctl disable networking.service这个耗时是有阈值的&#xff0c;一般大于15s的算&#x…

【数据结构与算法】8.二叉树的基本概念|前序遍历|中序遍历|后序遍历

&#x1f4da;博客主页&#xff1a;爱敲代码的小杨. ✨专栏&#xff1a;《Java SE语法》 | 《数据结构与算法》 | 《C生万物》 |《MySQL探索之旅》 |《Web世界探险家》 ❤️感谢大家点赞&#x1f44d;&#x1f3fb;收藏⭐评论✍&#x1f3fb;&#xff0c;您的三连就是我持续更…

vue封装请求、合并js、合并多个js

vue封装请求、合并js、合并多个js 作为一个后端开发&#xff0c;写前端时发现&#xff0c;每次导入api接口都会有一堆代码&#xff0c;像下面这样&#xff1a; import {footprintList, footprintDelete} from /api/userApi.js import {addressList} from /api/userApi.js impor…

设置Linux开发板开机自启动QT程序的报错解决办法

设置Linux开发板开机自启动QT程序报错解决办法 设置开发板开机自启动QT 打开 /etc/init.d/rsC 文件&#xff0c;添加以下内容 cd / ./my_start_run.shmy_start_run.sh 是自己编写的自启动脚本&#xff0c;内容例如下&#xff1a;(也可以将这些直接写到 /etc/init.d/rsC 文件…

【算法刷题 | 贪心算法02】4.24(摆动序列)

文章目录 3.摆动序列3.1题目3.2解法&#xff1a;贪心3.2.1贪心思路3.2.2代码实现 3.摆动序列 3.1题目 如果连续数字之间的差严格地在正数和负数之间交替&#xff0c;则数字序列称为 摆动序列 。 第一个差&#xff08;如果存在的话&#xff09;可能是正数或负数。仅有一个元素…

嵌入式总线协议基础教学

在嵌入式系统设计中&#xff0c;总线协议&#xff08;bus protocols&#xff09;扮演着至关重要的角色&#xff0c;它们定义了设备如何在共享通信路径上交换数据。 本文将介绍两种常见的嵌入式总线协议&#xff1a;IC&#xff08;Inter-Integrated Circuit&#xff09;和SPI&a…

BFS解决FloodFill算法:(Leetcode:733. 图像渲染)

题目链接&#xff1a;733. 图像渲染 - 力扣&#xff08;LeetCode&#xff09; 使用广度优先遍历算法解决该问题&#xff1a; 从初始位置开始搜索&#xff0c;初始位置符合条件就入栈&#xff0c;并修改初始位置值。初始位置出栈。 再从初始位置开始广度优先搜索&#xff08;…

IDM下载器安装cmd注册

一、下载注册 安装包去IDM官网下载最新的试用版即可 或者直达百度网盘下载&#xff08;担心被河蟹&#xff0c;放在txt中了&#xff09;包含IDM下载器安装包和注册软件 IDM下载器安装包和注册软件下载地址链接 https://download.csdn.net/download/qq_31237581/89215452 如果…

sscanf和scanf区别

sscandf 从字符串中提取数据 scanf 标准输入流读取数据 int num; sscanf("42", "%d", &num);float f; sscanf("3.14", "%f", &f);char str[20]; sscanf("Hello, World!", "%s", str);int a, b; sscanf(…

vue3 引入@tsparticles/vue3和@tsparticles/slim 实现粒子特效

1.安装&#xff1a; yarn add tsparticles/vue3 tsparticles/slim2.main.ts 引入 import Particles from "tsparticles/vue3"; import { loadSlim } from "tsparticles/slim";app.use(Particles as any, {init: async (engine: any) > {await loadSli…