Linux相关概念和易错知识点(16)(Shell原理、进程属性和环境变量表的联系)

Shell原理及其模拟实现

认识进程exec系列函数、命令行参数列表、环境变量之后,我们可以尝试理解一下Shell的原理,将各方知识串联起来,让Shell跑起来才能真正理解这些概念。我会以模拟Shell执行的原理模拟一个Shell。途中配上相关讲解。

1.Shell进程的创建

Shell是用户启动后系统为我们启动的第一个进程,用于启动CLI程序,为后面我们的命令行操作做准备,在Linux中,这个Shell具体程序是bash。

bash在/usr/bin/bash下,每一个用户登陆后都会先在内存中创建PCB,再从硬盘中将代码和数据读到内存中来,就和其他进程一样。

2.环境变量的配置

我们从上面那张图中就能发现,bash并不是在每个用户的home目录下,而是在公共区域。同一个程序却能生成针对不同用户登陆的bash,这和环境变量的配置有直接关系。当创建bash读取硬盘数据时,bash会从登陆用户的home目录下读取.bash_profile和.bashrc,里面有环境变量配置的相关信息。但是注意,配置文件里面并不包含所有的环境变量,它是局部的,还有一些环境变量是从bash的父进程继承下来的或是后面生成的。

下面是配置文件里面的一些内容,这些环境变量都会被bash导入到内存中形成一份临时环境变量表,在代码或命令行中修改环境变量就是修改的内存中的环境变量,只要不改硬盘文件,就不会修改默认环境变量。

环境变量是进程中最基础的部分,所有进程启动时都要导入一份环境变量

对于Shell进程,方式有:父进程继承、读取配置文件、后续自动生成(HOME -> cwd -> PWD)

对于其它父子进程而言方式有:父进程继承

当然我们都可以手动添加环境变量。

我们要模拟Shell进程,这意味着我们要完成父进程继承、读取配置文件、后续自动生成。这显然超出了我们的能力,把上面的实现了基本上我们就写出了一个系统。但我们可以模拟普通进程的导入环境变量,拷贝一份环境变量表做模拟。

这里有个易错点,一定要注意。main函数的三个参数(int argc, char** argv, char** env)的env是一个局部变量,且只有我们写int main(int argc, char** argv, char** env)时env才会被导入到函数栈帧中(写了参数调用的main函数就不一样了),后续如果要用env这个变量就要自己手动传参。env里面就存着环境变量。但程序中还有一个变量char** environ,它是全局的,且我们只要extern之后就可以在任何地方调用。

我们最终采用environ方案。

对于这个模拟的Shell程序来讲,我们手动显式实现了一个环境变量表,而我们需要搞清楚这个程序本身还有一个环境变量,那才是该程序真正的环境变量表,我们environ本质就是从该程序的本身的变量表拷贝过来的。

3.对环境变量表、命令行参数列表、进程属性之间的关系解读

在我们的模拟代码中,环境变量表存在于Shell程序的全局区域,可以被任意调用,在进程角度看来环境变量表属于代码和数据部分,本质上是存在于mm_struct管理的映射区域的,这和进程属性直接存在PCB中有本质区别。命令行参数argc和argv都是属于代码和数据的。

(1)对mm_struct进一步解释

进程的mm_struct里面管理进程地址空间及其映射的一部分物理空间,这部分空间里面存的是进程的代码和数据,而除了代码和数据是不会存在于进程地址空间里的(比如进程属性),因此我们在栈区、堆区等地方是找不到进程的属性struct mm_struct* mm或者是cwd的,因为它们是PCB的属性,直接存在物理内存中,没有页表映射。操作系统直接管理,用户永远无法也无需直接获取它们的地址。

综上,进程属性本质上是属于PCB,由Kernel直接管理的,我们永远拿不到它们的物理地址。而相比之下,环境变量属于进程的代码和数据,存在于mm_struct管理的物理空间,对应进程地址空间的位置是高于栈区的。

(2)环境变量PWD和进程属性cwd的关系

在这里我们需要了解一些后续生成的环境变量是如何来的。其中最重要的就是PWD和cwd的关系!

当读取配置文件后,HOME被配置好了,之后cwd会使用chdir修改自己的当前工作目录,再之后会借助cwd里面的数据使用putenv这个函数创建一个新的环境变量PWD。PWD环境变量是借助cwd这个进程属性来初始化的。

注意,只有Shell进程创建时会干这事。其余进程创建时都会使用写时拷贝的技术继承父进程的环境变量表,不会有任何读取文件或是自动添加环境变量操作。

3.打印命令行提示符

命令行提示符的格式是:[用户名(USER)@主机名(HOSTNAME) 当前路径(PWD)]$/#

这部分主要就是字符串处理相关的知识。获取的环境变量是从我们自己拷贝得到的环境变量表中取,而不是在该进程原本的环境变量表中取。

注意C/C++混编的情况下string有着避免野指针的优势,我们可以在函数里定义string,然后返回它,这样做不会出现和数组那样的野指针问题,会自动初始化一个新的string。当然我们要注意C/C++之间的转换,如c_str()这种接口要熟悉

4.获取并分析用户命令

这是一个非常容易犯错误的地方。全局的argv和argc、获取用户命令的CommandBuffer每次都要清空数据,之后用fgets安全读入一行,strtok分割字符串并存入argv中。

注意回车符也会被读入,读入后的下一个字符被标记为'\0',读取结束,因此我们要手动处理字符串中有回车的情况,将它改成'\0'。

5.执行命令

(1)子进程执行命令

借助execvp和argv,我们可以实现子进程执行程序,父进程wait子进程。这样做的好处是子进程执行失败完全不会影响父进程的安全性,如果直接让父进程执行所有命令,那么如果出了一个较严重的错误,父进程就直接被挂掉了,这显然不是我们希望看到的。

(2)内建命令

内建命令用于修改当前进程环境变量的值或者要访问只有该进程能访问的数据。如echo能访问本地变量,cd要修改本进程的环境变量。因为子进程执行指令没办法访问父进程的数据,进程之间的独立性决定了这类命令只能直接由父进程执行。

思路就是穷举法,将需要的命令手动在父进程处理。注意chdir修改的是进程属性cwd,进程属性和环境变量之间在进程运行时是各改个的,只有在Shell创建时的初始化时才存在关联。它们之间的同步需要手动维护。

我们从一个更加底层的角度上来想,进程属性属于PCB、系统直接管理对象,而环境变量是程序代码和数据的一部分。当修改一边时,另一边理所应当保持独立,只不过Shell内部维护导致我们大部分看上去是同步的,但事实上修改cwd后PWD依然不变,cwd和PWD都只是数据而已,并没什么大不了的,因此我们需要手动putenv。 

在有的时候会发现存在不同步的情况。比如在Shell进程内切换用户,环境变量表会变,因为切换用户会重新读取配置文件,但进程始终是同一个,进程创建者不变。

注意getenv和putenv我们都要自己实现,因为系统给我们的这些函数都只会到进程自带的环境变量表中查找,我们要实现到自己的环境变量中查找、添加等,都要自己写。

最后可以使用全局的lastcode存储退出码,同样使用内建命令处理echo来获得退出码,底层逻辑是一样的,也很简单,这里就不再讲述了。唯一需要注意的是当没有找到命令时,错误信息和退出码是要在exec函数后面更新的,exec函数执行成功就不会执行下面的语句,执行失败就要。

总结:

我们通过Shell的原理和模拟实现主要是为了搞清进程属性和环境变量表、命令行参数表的关系。我们发现,一个进程 = PCB + 代码和数据,PCB可以管理这些代码和数据。其中进程属性直接是PCB的成员,被系统直接管理。而环境变量表、退出码、命令行参数列表本质都是存在代码和数据中的,也可以叫程序的上下文中。正是这样的差异导致了环境变量和进程属性的独立性,也帮助我们理解环境变量表是如何传给子进程的,命令行参数从读取到传给argv的过程是怎样的,以及退出码是如何读取到的。整个Shell的知识点都被串联起来了,很值得我们消化。

全部代码:


#include <sys/wait.h>
#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <string>
#include <unistd.h>
using namespace std;const int basenum = 100;
const int basesize = 1024;//命令行参数列表和环境变量表放在Shell的代码和数据中
char* genv[basenum];
char* gargv[basesize];
int gargc;//存储即将更新的PWD,维护环境变量的更新,保持cwd和PWD的同步
int posPWD = 0;//随时存储PWD的位置,方便后续找到
char newPWD[basesize];int lastcode;//存储退出码void InitEnv()//初始化环境变量表,一般子进程是这样继承的,Shell进程是从文件读配置文件的,这里简略一点
{extern char** environ;for(int curi = 0; environ[curi]; curi++){genv[curi] = (char*)malloc(strlen(environ[curi]) + 1);//不用sizeof,它会按指针大小算memcpy(genv[curi], environ[curi], strlen(environ[curi]) + 1);}
}string GetEnv(string oneEnv)//用string防止不必要全局字符串,值拷贝避免野指针
{for(size_t i = 0; i < basenum; i++){for(size_t j = 0; j < oneEnv.size(); j++){if(oneEnv[j] != genv[i][j])//有不一样的就说明不匹配break;if(j == oneEnv.size() - 1)//全部相等{string User;for(size_t k = oneEnv.size() + 1; genv[i][k]; k++)User += genv[i][k];posPWD = i;//存储PWD信息,方便后续找到return User;}}}return nullptr;
}void PrintCommandLine()
{string GetUser = GetEnv("USER");string GetPwd = GetEnv("PWD");string GetHostName = GetEnv("HOSTNAME");if((GetPwd = string(GetPwd.begin() + GetPwd.rfind('/'), GetPwd.end())).size() > 1)//表达式的返回值是string,如果返回值不是根目录,都要把/删掉GetPwd.erase(0, 1);printf("[%s@%s %s]%c ", GetUser.c_str(), GetHostName.c_str(), GetPwd.c_str(), GetUser == "root" ? '#' : '$');
}void GetCommandLine(char* CommandBuffer)//使用值拷贝string防止出现野指针
{memset(CommandBuffer, '\0', basesize);//每次都初始化读取用的字符串fgets(CommandBuffer, basesize, stdin);CommandBuffer[strlen(CommandBuffer) - 1] = '\0';
}void ParseCommandLine(char* CommandBuffer)
{gargc = 0;//每次解析命令前先把上一个命令的信息删除memset(gargv, '\0', basesize);if(strlen(CommandBuffer) == 0)return;gargv[gargc++] = strtok(CommandBuffer, " ");while((bool)(gargv[gargc++] = strtok(nullptr, " ")));gargc--;
}void ExecuteCommand()//没有指令的情况走不到这个函数
{pid_t ret = fork();if(ret == 0){execvpe(gargv[0], gargv, genv);printf("-bash: %s: command not found\n", gargv[0]);lastcode = 1;}else{int status = 0;waitpid(ret, &status, 0);if(!WIFEXITED(status))lastcode =  1;elselastcode = WEXITSTATUS(status);}
}bool CheckAndExecBuiltCommand()
{if(gargc == 0)//如果没有指令,直接进行下一轮循环,通过return true省的走下一个函数return true;if(strcmp("cd", gargv[0]) == 0){if(gargc == 2){memset(newPWD, '\0', basesize);chdir(gargv[1]);snprintf(newPWD, basesize, "PWD=%s", gargv[1]);genv[(GetEnv("PWD"), posPWD)] = newPWD;//每次调用GetEnv都会刷新posPWD的位置,用逗号表达式的特性实现lastcode = 0;return true;}return false;}if(strcmp("export", gargv[0]) == 0){if(gargc == 2){int curi = 0;while(genv[curi]) curi++;genv[curi] = (char*)malloc(strlen(gargv[1]) + 1);memcpy(genv[curi], gargv[1], strlen(gargv[1]) + 1);lastcode = 0;return true;}return false;}if(strcmp("env", gargv[0]) == 0){if(gargc == 1){for(int i = 0; genv[i]; i++)printf("%s\n", genv[i]);lastcode = 0;return true;}return false;}if(strcmp("echo", gargv[0]) == 0){if(gargc == 2 && strcmp("$?", gargv[1]) == 0)printf("%d\n", lastcode);lastcode = 0;return true;}return false;
}int main()
{char CommandBuffer[basesize] = { 0 };InitEnv();while(true){PrintCommandLine();//打印命令行提示符GetCommandLine(CommandBuffer);//读取命令ParseCommandLine(CommandBuffer);//解析命令至argc和argv中if(CheckAndExecBuiltCommand()) continue;ExecuteCommand();}return 0;
}

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

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

相关文章

基于flask和neo4j的医疗知识图谱展示问答系统

如果你仍在为毕业设计的选题发愁&#xff0c;或者想通过技术项目提升专业实力&#xff0c;这个基于Flask和Neo4j的医疗知识图谱展示与问答系统&#xff0c;绝对是个不错的选择&#xff01; 项目亮点大揭秘&#xff1a; 知识图谱与问答结合&#xff1a;我们采用了医疗场景下的知…

时间数据可视化基础实验(南丁格尔玫瑰图)——Python热狗大胃王比赛数据集

【实验名称】 实验一&#xff1a;时间数据的可视化 【实验目的】 1.掌握时间数据在大数据中的应用 2.掌握时间数据可视化图表表示 3. 利用python程序实现堆叠柱形图的可视化 【实验原理】 时间是一个非常重要的维度与属性。时间序列数据存在于社会的各个领域&#xff0c;如&a…

【Power Query】List.Select 筛选列表

List.Select 筛选列表 ——在列表中返回满足条件的元素 List.Select(列表,判断条件) 不是列表的可以转成列表再筛选&#xff0c;例如 Record.ToList 不同场景的判断条件参考写法 (1)单条件筛选 列表中小于50的数字 List.Select({1,99,8,98,5},each _<50) (2)多条件筛…

高低温探针台的工作原理及其构造

高低温探针台是一种用于材料科学、物理、化学等领域的实验设备&#xff0c;主要用于在高温和低温环境下对材料进行各种实验和研究。 高低温探针台的工作原理是将样品放置在加热和冷却组件上&#xff0c;然后使用各种测量仪器对其进行实验和测量。具体来说&#xff0c;其工作流程…

NVR批量管理软件/平台EasyNVR多个NVR同时管理支持UDP和TCP传输协议

随着科技的飞速发展&#xff0c;视频技术已成为现代社会不可或缺的一部分&#xff0c;广泛应用于安防监控、娱乐传播、在线教育、电商直播等多个领域。在这一背景下&#xff0c;NVR管理平台EasyNVR作为一款高效、灵活的视频监控管理系统&#xff0c;正经历着前所未有的发展机遇…

iOS 本地存储地址(位置)

前言: UserDefaults 存在沙盒的 Library --> Preferences--> .plist文件 CoreData 存在沙盒的 Library --> Application Support--> xx.sqlite 一个小型数据库里 (注:Application Support 这个文件夹已开始是没有的,只有当你写了存储代码,运行之后,目录里才会出…

MT-Pref数据集:包含18种语言的18k实例,涵盖多个领域。实验表明它能有效提升Tower模型在WMT23和FLORES基准测试中的翻译质量。

2024-10-10&#xff0c;由电信研究所、里斯本大学等联合创建MT-Pref数据集&#xff0c;它包含18种语言方向的18k实例&#xff0c;覆盖了2022年后的多个领域文本。通过在WMT23和FLORES基准测试上的实验&#xff0c;我们展示了使用MT-Pref数据集对Tower模型进行对齐可以显著提高翻…

C++list

1.迭代器的功能以及性质 功能有iterator&#xff0c;reverse_iterator&#xff0c;const_iterator&#xff0c;const_reverse_iterator 性质有单向&#xff0c;双向和随机&#xff0c;单向的迭代器只能操作&#xff0c;双向只能/--俩种&#xff0c;随机则可以执行/--//-的操作…

【C++篇】栈的层叠与队列的流动:在 STL 的韵律中探寻数据结构的优雅之舞

文章目录 C 栈与队列详解&#xff1a;基础与进阶应用前言第一章&#xff1a;栈的介绍与使用1.1 栈的介绍1.2 栈的使用1.2.1 最小栈1.2.2 示例与输出 1.3 栈的模拟实现 第二章&#xff1a;队列的介绍与使用2.1 队列的介绍2.2 队列的使用2.2.1 示例与输出 2.3 队列的模拟实现2.3.…

[快速阅读八] Matlab中bwlookup的实现及其在计算二值图像的欧拉数、面积及其他morph变形中的应用。...

以前看过matlab的bwlookup函数&#xff0c;但是总感觉有点神秘&#xff0c;一直没有去仔细分析&#xff0c;最近在分析计算二值图像的欧拉数时&#xff0c;发现自己写的代码和matlab的总是对不少&#xff0c;于是又去翻了下matlab的源代码&#xff0c;看到了matlab里实现欧拉数…

JS轮播图实现自动轮播、悬浮停止轮播、点击切换,下方指示器与图片联动效果

代码&#xff1a; <!DOCTYPE html> <html lang"zh-CN"> <head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title>Document</title><s…

计算机网络原理总结C-网络层

网络层 网络层提供的两种服务网际协议IP 虚拟互连网络IP地址子网掩码&#xff08;无分类编址CIDR&#xff09;IP地址和MAC地址IP数据报格式&#xff08;路由&#xff09;转发分组的流程 因特网的路由选择协议&#xff08;动态路由协议&#xff09; 网际控制报文协议ICMPIP多播…

麒麟v10 arm64 部署 kubesphere 3.4 修改记录

arm64环境&#xff0c;默认安装 kubesphere 3.4 &#xff0c;需要修改几个地方的镜像&#xff0c;并且会出现日志无法显示 1 fluentbit:v1.9.4 报错 <jemalloc>: Unsupported system page size Error in GnuTLS initialization: ASN1 parser: Element was not found. &…

Java最全面试题->数据库/中间件->Redis面试题

文章目录 Redisredis的数据类型有哪些?Redis 内部结构有哪些?Memcache 与 Redis 的区别都有哪些?为什么 redis 需要把所有数据放到内存中?Redis 如何进行持久化?RDB和AOF的优缺点有哪些?什么是缓存穿透?如何避免?什么是缓存雪崩?何如避免?redis的淘汰策略有哪些?Red…

用更多的钱买电脑而不是手机

如果&#xff0c;我们对自己的定义是知识工作者&#xff0c;那么在工作、学习相关的电子设备投入上&#xff0c;真的别舍不得花钱。 需要留意的是&#xff0c;手机&#xff0c;对于大部分在电脑前工作的人&#xff0c;不是工作设备。在我看来&#xff0c;每年投入到电脑的钱&…

【高级IO】IO多路转接之select

select函数 我们知道IO 等待 拷贝&#xff0c;系统提供select函数来实现多路复用输入/输出模型&#xff1b;select系统调用是用来让我们的程序监视多个文件描述符的状态变化的&#xff1b;程序会停在select这里等待&#xff0c;直到被监视的文件描述符有一个或多个发生了状态…

植物健康,Spring Boot来助力

3系统分析 3.1可行性分析 通过对本植物健康系统实行的目的初步调查和分析&#xff0c;提出可行性方案并对其一一进行论证。我们在这里主要从技术可行性、经济可行性、操作可行性等方面进行分析。 3.1.1技术可行性 本植物健康系统采用SSM框架&#xff0c;JAVA作为开发语言&#…

钉钉录播抓取视频

爬取钉钉视频 免责声明 此脚本仅供学习参考&#xff0c;切勿违法使用下载他人资源进行售卖&#xff0c;本人不但任何责任! 仓库地址: GItee 源码仓库 执行顺序 poxyM3u8开启代理getM3u8url用于获取m3u8文件userAgent随机请求头downVideo|downVideoThreadTqdm单线程下载和…

【纯血鸿蒙】HarmonyOS和OpenHarmony 的区别

一、开源鸿蒙&#xff08;Open Harmony&#xff09; 鸿蒙系统愿来的设计初衷&#xff0c;就是让所有设备都可以运行一个系统&#xff0c;但是每个设备的运算能力和功能都不同&#xff0c;所以内核的设计上&#xff0c;采用了微内核的设计&#xff0c;除了最基础的功能放在内核…

logback 如何将日志输出到文件

如何作 将日志输出到文件需要使用 RollingFileAppender&#xff0c;该 Appender 必须定义 rollingPolicy &#xff0c;另外 rollingPollicy 下必须定义 fileNamePattern 和 encoder <appender name"fileAppender" class"ch.qos.logback.core.rolling.Rollin…