Linux·环境变量与进程地址空间

news2024/10/4 21:57:08

1. 命令行参数

        各位可能见过main函数也是有参数的,只是我们平时写的代码都比较简单,用不到main函数的参数,下面我们看一下main函数的参数是什么又是怎么用的

        我们看这样一段代码

                

        其编译运行后的效果是这样的

                

        我们将main函数后面的那两个参数叫命令行参数列表,argc 表示参数个数,argv[ ] 数组表示参数的清单。        

        我们对代码稍加改造

                        

        就可以达到这样的效果

        同一个程序可以根据命令行参数,根据选项的不同,表现出不同的功能,就比如 ls -a, ls -al 这种命令。

        main函数的命令行参数传递原理大概就是:我们输入的命令是一串字符串,这串字符串先由shell拿到,然后按空格打散形成一张表(argv[]),和元素个数(argc),argv中每个元素都是一个指针指向一个字符串这个效果我们在第一次的实验中也见到了。

2. 环境变量

        main函数的参数还有一个env[ ] 表这张表是该进程的环境变量表,我们先见一见这个表

                        

        env表的最后是空,因此循环结束条件这么写没问题。

        我们编译运行,出来的这24行东西就是这个 ./code 进程的环境变量。

        所有环境变量的格式都是key=value,也就是数字对应属性信息。这些环境变量中有部分是我们认识的,比如3号的SHELL=bin/bash shell的版本是bin/bash的,7号的当前用户,9号一大堆的ls配色方案,12号pwd

        环境变量表都是继承自父进程的,每个进程都有环境变量表。

2.1 几个环境变量

        下面我们感受几个环境变量的作用效果

        如果想查环境变量还可以使用指令 env

2.1.1 PATH

        我们知道,ls 虽然作为系统命令,但是说白了它也是个可执行程序,但是它跟我们自己写的可执行程序又有点不一样,自己写的程序运行时必须要在程序名前加上 ./ 但是 ls 命令却不用。

        这个问题其实我们之前解释过,因为想要执行一个程序,比如 ls 就要让系统知道这个程序的路径是什么,这样才能通过这个路径启动程序,因此自己写的程序要加一个 ./ 表示我要执行的程序就在当前目录下。但是 ls 不用指定路径是因为系统知道要去 /usr/bin 目录下找这个程序,如果我们把自己写的程序放到 /usr/bin 中后也可以不用 ./ 指定路径了,就像系统指令一样执行程序了。

        

        前面说 env指令 可以查所有环境变量,如果要查特定环境变量 echo $PATH 指令

        这样查出来的信息就是PATH指令特定的环境变量,其中每条路径都由分号 : 隔开。

        PATH环境变量的作用就是告诉shell应该去哪些路径下查指令,当shell要执行一个命令行命令的时候,它首先就会按顺序去这些由冒号 : 分割开的多个子路径下查找对应的命令。

        也就是说PATH是系统可执行文件的搜索路径集合。也就是说,我们现在有两种方案来达到不带路径就可以执行程序的效果。1.将程序拷贝到上述任意一条路径下,2.将自己的路径添加到PATH环境变量中去。

        添加的命令就是 PATH=$PATH : 新增路径

        $PATH就是老PATH作为变量放在那里,避免我们再将老PATH敲一遍

        环境变量是内存级别的变量,即使我们这样更改了之后在重启终端之后新增的环境变量更改都会消失。如果说非要保存对环境变量的更改,可以尝试到家目录下vim进这两个文件中更改。

        这个是系统的配置文件,每次启动一个终端,申请操作系统服务的时候,操作系统都会先从这些配置文件中提取到环境变量,如果在这两个文件中修改的话就相当于是在磁盘级别的修改,是可以保存修改的了。

2.1.2 HOME 与 工作目录

        我们可以echo查home环境变量的信息,效果就是展示当前用户的家目录

                

        当用户登录的时候系统要创建对应的bash,而准备bash的时候要读取相关环境变量的配置文件,在此时用户对应的家目录就被写进了HOME环境变量中,因此我们在登录进机器后就直接处在自己的家目录下。

        bash作为一个进程,一定有自己的cwd也就是当前工作目录,我们可以查看一下

        先查bash进程的pid再进入proc目录查看bash进程的cwd发现此时工作目录就是/proc/22574,再回到家目录下重新查看bash进程的当前工作目录,发现变更成了家目录。也就是说bash的当前工作目录是根据当前所在目录变化的,而由于子进程的环境变量又是从父进程继承过来的,因此一般情况下我们运行程序的时候它的工作目录就是它所在的目录。但实际上一个程序的工作目录是取决于,bash的当前工作做目录,而bash的工作目录又取决于当前所在的目录。也就是说如果我们在家目录下用绝对路径启动一个进程,那该进程的工作目录就是家目录。

2.1.3 SHLL

        这个环境变量会记录下来用户登录的时候启动的是哪一个shell,由此会把shell相关的一些可执行程序的路径和程序记录下来,就比如我们现在shell的版本是bash

2.3.4 PWD 与 代码中获取环境变量

        保存当前进程所在工作路径

                

        在程序外我们可以查看PWD环境变量得知当前进程的工作路径,那在代码中获知进程的工作路径也是有必要的。之前我们的获取方案是main函数参数获取环境变量 env[ i ] ,其实还有一种更简单的方案:使用系统调用 char* getenv(const char* )

        ​​​​​​​        

        我们使用一下这个系统调用

        ​​​​​​​                

        其效果就是这样

        可以看到其效果也验证了我们之前在home中所说的工作目录的问题,果然是在哪个目录下打开的程序,那它的工作路径就在哪里。

2.3.5 USER

        USER是当前用户的环境变量

        ​​​​​​​        

        可以看到我用atlanteep的身份和root的身份去运行同一份程序时运行的效果是不一样的,通过这个效果就可以根据环境变量写出控制权限的程序。

2.3.6 OLDPWD 与 cd -

        cd - 命令可以返回上次打开的目录

        这个功能的实现就是通过环境变OLDPWD实现的,这个环境变量会记录下来上次所在的路径

3. 本地变量

        除了环境变量之外在Linux中我们还可以设置本地变量,本地变量与环境变量一样,都是内存级别的变量,重启bash进程之后之前的修改就都没了。

        ​​​​​​​        

        查看本地变量使用 set 命令

        通过 set命令 我们不仅可以看见环境变量,还可以看到我们刚才自己定义的本地变量。

        我们还可以使用 explort命令 将本地变量变成环境变量

        事实上,我们启动机器之后,首先会从磁盘中把OS加载到内存中去,之后用户登录,操作系统就会给该用户分配一个bash命令行解释器进程,那么bash进程会维护3张表,argv[](命令行参数表) env[](环境变量表) 本地变量表

        我们explort就是将变量直接从本地变量表迁移到环境变量表中。

        如果想要移除某个环境变量使用 unset命令

        这样刚才那个a就没有了

        本地变量一般的作用是在运维的时候写一些自动化的脚本的。环境变量是可以被子进程继承下去的,但是本地变量不能被子进程继承。

        环境变量因为有这种继承的特性,因此我们称全局变量具有全局属性

        为什么要让环境变量具有全局属性呢?第一:环境变量是系统的配置信息,尤其是有指导性的一些功能,比如当前工作目录,它是系统配置起效的一种表现。第二:进程虽然具有独立性,但是可以通过环境变量来进行进程间传递数据。

3.1 一种新方案访问环境变量

        我们前面介绍了访问环境变量的两种方案,接下来我们再介绍第三种 environ全局变量 访问环境变量

        它指向环境变量表,因此其类型是 char** 的二级指针类型。

        因此我们可以通过这个指针来获取环境变量

        ​​​​​​​        ​​​​​​​        

        可以看到,通过全局变量environ也可以获取环境变量

4. 进程地址空间

        我们先看一个现象

        我们定义一个全局变量gval,之后从父进程中分出一个子进程,两个进程同时打印各自的信息,同时子进程会修改gval的值。

        因为进程间具有独立性,因此子进程改变全局变量的大小,但是父进程gval的值还是100,这些都符合我们的认知。但是观察gval的地址在子进程和父进程都是一样的,这是没道理的,同一块物理内存上是不可能存两份不同数据的。因此这说明之前我们在 c/c++ 中学到的 &取地址,取出的并不是真正的物理内存地址,这个取出来的地址实际上叫做 虚拟地址 或 线性地址 

        内存空间的地址(虚拟地址)分配如下图

        ​​​​​​​        ​​​​​​​        ​​​​​​​        

        由低地址到高地址分别是,正文代码、初始化全局数据,未初始化全局数据、堆、栈、命令行参数环境变量,但是堆和栈之间还有一块很大的空间是共享区。

        上图的空间分配方案应该叫进程地址空间,它是逻辑上存在的,而不是真实的物理内存的分配方案。

        这个进程地址空间存在的意义是为了让每一个进程都认为自己是独占机器物理内存的大小,进程之间彼此不知道对方的存在,从而实现一定程度上的隔离和便于管理。

        形象一点讲,这个进程虚拟地址空间是操作系统给进程话的饼,告诉进程:你那些数据我都如上图那样放进物理内存了,你要用随时可以来取用。但其实物理内存中根本就不是那么存放的。既然操作系统给进程画了饼,那肯定也会给进程一个兑换饼的凭证,使进程可以用这个凭证来从逻辑上存在的内存地址中向OS申请,取到对应数据。

        这个凭证具体来说是一个内核数据结构对象,针对Linux操作系统来讲就是struct mm_struct进程地址空间

        pcb中会存储这个凭证的指针,后面可以通过这个凭证向OS索要物理内存的使用权。

        mm_struct中描述的就是进程地址空间的逻辑,那进程地址空间又是由各种数据段组成的,其区域的划分就是用某区域的起始地址和结束地址,存放在 mm_struct 中用来给进程地址空间做变量存放地址的指导,注意还是那句话这里划分出来的区域也是虚拟的并不是给真正的物理内存划分。

        在mm_struct创建的时候其中具体哪一块空间要开多大都是在可执行程序中写好的,但是像栈区堆区这种运行中使用的地址空间就是在运行中申请的。

        struct mm_struct作为虚拟地址存放表,依靠页表将虚拟地址映射到物理内存的真实地址中。

        当我们创建一个子进程的时候,不仅仅要从父进程那里把PCB继承过来,还要把mm_struct虚拟地址空间继承过来,接着再把父进程的页表继承过来。

        也就是说在不修改变量的时候,子进程的虚拟地址空间、页表、甚至页表映射的物理内存与父进程都是完全一致的,也就是说它们的数据,代码全都是共享的。

        但是如果此时子进程(或父进程),对于某个数据进行了更改,此时操作系统就会通过该变量的虚拟地址找到真实物理地址,然后将物理地址中的内容拷贝一份再开一份新空间,将新数据放到物理内存中,然后把页表中的所映射的物理内存更新,但是虚拟地址不变。这就是为什么我们一开始看到父进程和子进程的全局变量地址相同但数据不同的原因,因为我们看到的地址只是虚拟地址,但实际上子进程的相同虚拟地址所映射的真实物理地址却与父进程不同了。这种操作叫做OS的写时拷贝机制

4.1 页表

        我们这里只是粗谈一下页表做一下铺垫,之后有章节详说

        页表的结构并没有我们刚才画的那么简单,并不是简单的提供虚拟地址映射物理内存的路径,它里面还有一些标志位,这次先说两个 rwx isexists

        ​​​​​​​        ​​​​​​​        

        rwx标志很简单,就是看这个变量有没有权限读写内存,如果没有权限读写却硬要写,那OS就直接杀进程。比如一个字符串常量,代码跑到了要修改这个常量的地方,要修改就要访问内存,直接查看页表,发现没有写权限,那就拒绝写入,杀进程。这个拒绝写入的事情是操作系统在做,因此编译器检查不出来错误,但是为了方便程序员自查,于是编译器提供const关键字来提示程序原一个变量在内存中是否有写权限。

        isexists标志表示一个变量是否真正加载到物理内存当中去了。比如一个黑猴有130G但是我电脑的内存只有4G那玩的时候不可能把整个黑猴都加载进内存,只能一部分一部分的加载,比如开局剧情跑完了,之后很长一段时间肯定都不会用到这段代码了,那就将这段代码踢出内存并在这个标志位标注,或者后面很久才会用到的代码也不会加载进内存。这种操作叫分批加载,或者分批挂起。

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

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

相关文章

排序算法剖析

文章目录 排序算法浅谈参考资料评价指标可视化工具概览 插入排序折半插入排序希尔排序冒泡排序快速排序简单选择排序堆排序归并排序基数排序 排序算法浅谈 参考资料 数据结构与算法 评价指标 稳定性:两个相同的关键字排序过后相对位置不发生变化时间复杂度空间复…

MyBatisPlus——学习笔记

MyBatisPlus 一、导入依赖 <!-- MyBatisPlus --><dependency><groupId>com.baomidou</groupId><artifactId>mybatis-plus-boot-starter</artifactId><version>3.5.2</version></dependency><!-- MySql --><de…

C++基础(7)——STL简介及string类

目录 1.STL简介 1.1什么是 1.2STL的历史版本 1.3STL的六大组件 ​编辑 1.4有用的网址 2.string类 2.1string的多种定义方式 2.2string的插入 2.2.1尾插&#xff08;push_back&#xff09; 2.2.2insert插入 2.3拼接&#xff08;append&#xff09; 2.4删除 2.4.1尾…

CoRL 2024 麻省理工学院提出T3触觉Transformer,打破触觉感知的壁垒,重塑未来机器人

在智能机器人领域&#xff0c;触觉感知的研究正逐渐成为关注的焦点。然而&#xff0c;如何让机器人通过触觉更智能地感知和操作&#xff0c;依然是一个未解决的挑战。基于相机的触觉感知是一种通过在软弹性体下嵌入相机来捕获与环境的细粒度交互的感知方法&#xff0c;是最流行…

Java报错输出的信息究竟是什么?

Java报错输出的信息究竟是什么&#xff1f; 本篇会带大家了解一下java运行时报错输出的信息内容&#xff0c;简单学习一下虚拟机内存中Java虚拟机栈的工作方式以及栈帧中所存储的信息内容 异常信息 当你的程序运行报错时&#xff0c;你是否会好奇打印出来的那一大坨红色的究竟…

搜索引擎相关的一段实习经历

0 前言 就是跟搜索相关的一段经历。主要工作就是建立倒排索引库相关的一些简单内容。 又翻到了以前的工作&#xff0c;权作纪念。 就是简单的封装cpp的库供python语言调用。 反正就是很多版本问题等等吧各种鬼问题。 我感觉这个思路可能还是待考证。 跨语言的调用我感觉还是不…

泛型编程--模板【C++提升】(特化、类属、参数包的展开、static、模板机制、重载......你想知道的全都有)

更多精彩内容..... &#x1f389;❤️播主の主页✨&#x1f618; Stark、-CSDN博客 本文所在专栏&#xff1a; C系列语法知识_Stark、的博客-CSDN博客 其它专栏&#xff1a; 数据结构与算法_Stark、的博客-CSDN博客 C系列项目实战_Stark、的博客-CSDN博客 座右铭&#xff1a;梦…

Java中的while和do...while循环

while和do...while循环 while循环基本语法执行流程注意事项练习 do...while循环基本语法说明流程图练习 while循环 基本语法 循环变量初始化; while(循环条件){循环体&#xff08;语句&#xff09;;循环变量迭代; }1&#xff09;while循环也有四要素&#xff1a;循环变量初始…

【JNI】普通类型的基本使用

简单使用 在上一期我们介绍了JNI的基本使用&#xff0c;这里简单介绍一下普通类型 HelloJNI.java&#xff1a;这里计算两个整型数的平均值&#xff0c;返回值类型为double public class HelloJNI { static {System.loadLibrary("hello"); }private native String …

electron-builder 首次执行报错问题解决

假日想研究一下 react electron 的使用&#xff0c;结果发现首次打包疯狂报错&#xff0c;研究了一下之后才发现是第一次的话 electron-builder 会从外面下载依赖包到我们系统中&#xff0c;由于某种力量导致压缩包无法下载或者是下载过慢导致失败&#xff0c;要解决其实也简单…

认知战认知作战:2024年9月30日中国股市大涨背景下的认知战分析报告

认知战认知作战&#xff1a;2024年9月30日中国股市大涨背景下的认知战分析报告 关键词&#xff1a;认知战, 中国股市, 信息操纵, 心理战, 技术战, 信息监管, 投资者素养, 国际合作, 法律法规, 协同作战, 谣言澄清, 市场情绪,认知作战,新质生产力,人类命运共同体,认知战,认知域…

《Linux从小白到高手》理论篇:深入理解Linux的计划任务/定时任务

值此国庆佳节&#xff0c;深宅家中&#xff0c;闲来无事&#xff0c;就多写几篇博文。本篇详细深入介绍Linux的计划任务/定时计划。 Linux的计划任务 在很多时候为了自动化管理系统&#xff0c;我们都会用到计划任务&#xff0c;比如关机&#xff0c;重启&#xff0c;备份之类…

二叉树--堆

1.二叉树的顺序结构 普通的二叉树是不适合用数组来存储的&#xff0c;因为可能会存在大量的空间浪费。而完全二叉树更适合使用顺序结 构存储。现实中我们通常把堆(一种二叉树)使用顺序结构的数组来存储&#xff0c;需要注意的是这里的堆和操作系统 虚拟进程地址空间中的堆是两…

CSP-J Day 4 模拟赛补题报告

姓名&#xff1a;王胤皓&#xff0c;校区&#xff1a;和谐校区&#xff0c;考试时间&#xff1a; 2024 2024 2024 年 10 10 10 月 4 4 4 日 9 : 00 : 00 9:00:00 9:00:00~ 12 : 30 : 00 12:30:00 12:30:00&#xff0c;学号&#xff1a; S 07738 S07738 S07738 请关注作者的…

Windows应用开发-解析AVI视频文件

本Windows应用解析AVI视频文件&#xff0c;以表格的方式显示AVI文件结构。并可以将结果保存到bmp图片。下面是&#xff0c;使用该应用解析一部AVI电影获得的图片。 应用开发信息 定义一个INFO结构&#xff0c;包含两个字符串对象&#xff0c;一个ULONGLONG变量&#xff0c;和…

奔驰AMG GT50升级原厂阀门运动排气声浪效果

AMG 排气系统 声浪级别可控制的AMG高性能排气系统可带来不同凡响的听觉体验。借助可调式废气风门&#xff0c;按下按钮&#xff0c;即可按需改变车辆的声浪&#xff0c;体验不同音色。静谧深沉或动感澎湃&#xff0c;悦耳声浪&#xff0c;如你所愿。

Python画笔案例-076 绘制纯画笔弹球

1、绘制纯画笔弹球 通过 python 的turtle 库绘制 纯画笔弹球,如下图: 2、实现代码 绘制纯画笔弹球,以下为实现代码: """纯画笔弹球动画.py读者可以在此基础上把它修改成一个拦球游戏。步骤为,建立一个Rect类,即矩形类。然后采用按键检测,当按了键时重画…

​​Python+Matplotlib可视化简单反函数和复合函数

import numpy as np import matplotlib.pyplot as plt# 设置中文字体 plt.rcParams[font.sans-serif] [SimHei] # 用黑体显示中文 plt.rcParams[axes.unicode_minus] False # 正常显示负号# 创建图形和子图 fig, (ax1, ax2) plt.subplots(1, 2, figsize(15, 6))# 反函数示…

Line: 折线图

对北京市、天津市、上海市、重庆市的近10年人口&#xff0c;做出折线图&#xff0c;效果 参考&#xff1a;Line - Basic_line_chart - Document (pyecharts.org) 1、折线图模板 import pyecharts.options as opts from pyecharts.charts import Linex_data ["Mon"…

基于Springboot+Vue的中医院问诊系统的设计与实现 (含源码数据库)

1.开发环境 开发系统:Windows10/11 架构模式:MVC/前后端分离 JDK版本: Java JDK1.8 开发工具:IDEA 数据库版本: mysql5.7或8.0 数据库可视化工具: navicat 服务器: SpringBoot自带 apache tomcat 主要技术: Java,Springboot,mybatis,mysql,vue 2.视频演示地址 3.功能 系统中…