文章目录:
1.环境变量
1.1 基本概念
环境变量一般是指操作系统中用来指定操作系统运行环境的一些参数
常见的环境变量
• PATH:保存了可执行程序的搜索路径
• SHELL:保存了命令行解释器的名称
• HOME:保存了当前用户的家目录
查看环境变量
• echo $[环境变量名称]:查看环境变量中的内容
• env:查看当前操作系统中所有的环境变量
1.2 设置环境变量
固定范式:export [环境变量名称] = $[环境变量名称]:[新增加的路径]
临时生效
执行在了命令行当中,只在当前终端生效
在当前终端测试如下图所示
新开一个终端测试
永久生效
~/.bashrc 或者 ~/.bash_profile:环境变量文件,修改此处只是写到了磁盘中
当新打开一个终端时,就会读取环境变量文件当中的内容,所以我们可以将设置的环境变量写到环境变量文件当中,这样每次启动终端的时候都会被加载
source ~/.bashrc 或者 source ~/.bash_profile 重新把文件加载一遍
测试如下:
打开一个新的终端测试:
1.3 获得环境变量值的三种方法
通过main函数的参数获取环境变量的值
命令行参数的个数也将可执行程序计算在内
通过main函数的参数获取环境变量值方法如下:
• argc :命令行参数的个数
• argv:指针数组,数组的每一个元素都是一个char*
含义:具体的命令行个数
• env:中保存的是每一条的环境变量
char* env[]:组织格式如下图
永远都在数组的最后一个元素存放一个NULL指针,可以作为条件判断环境变量的个数
通过getenv函数获取特定环境变量的值
• char* getenv(const char* name)
• name:环境变量名称
• 返回值:环境变量的值
下面我们以PATH为例
先通过echo $ PATH查看PATH的环境变量信息
接下来使用getenv函数获取PATH的环境变量的值
对比可得echo $PATH获得的环境变量的值和getenv函数获取的PATH环境变量的值是一样的
通过environ参数获取环境变量的值
environ:参数是C库当中定义的,程序员要使用可以使用extern关键字进行使用,类型是:char**
还是使用环境变量的组织格式控制环境变量的个数
2. 进程虚拟地址空间
程序员在代码中看到的地址并不是物理内存地址,而是操作系统内核虚拟出来的地址(进程的虚拟地址空间),虚拟地址并不能保存数据,而真正保存数据的是物理内存
使用fork()创建一个子进程,放一个全局变量g_val=10,会发现父子进程输出的值和地址都是一样的,这可以理解为,因为子进程是以父进程为模板拷贝的,父子进程并没有对变量进行修改
同一变量,地址相同其实是虚拟地址相同,,内容不同,其实变量的内容是在不同的物理地址里
但如果对代码做如下改动,会发现父子进程输出的变量值不一样,但地址是一样的,这证明了我们上面所说的,我们看到的地址不是物理内存地址,而是虚拟地址
如下图所示
进程虚拟地址空间构成了进程的独立性,一个进程修改数据,不会影响另外一个进程,如果变量值不同操作系统内核会在内存中重新开辟一个空间,用于存放变量
3. 如何通过虚拟地址找到物理内存地址呢?
通过上面我们了解到我们看到的地址都是虚拟地址,而真正保存数据的是在物理内存中的,而我们如何才能通过虚拟地址找到存储数据的物理内存的地址呢?数据在物理内存中又是如何存储的?
3.1 数据在物理内存中的存储
假设我们有一个8M大小的物理内存空间,有三个需要存储的数据大小分别为4M,2M,4M,有两种存储方式:
• 顺序存储:顺序存储又会出现如下图的两种情况:
第一种情况第二个4M的数据没办法存储,只能等第一个4M的数据释
放后存储
第二种情况只能等2M大小的数据释放后存储,或4M大小的数据释放后存储
• 离散存储 (推荐使用)
我们可以将物理内存分成不同的小块,然后进行离散存储,如下图所示存储
3.3 分页式
将虚拟地址(4G)分成一页一页的小块,将物理地址分成一页一页的小块(快的大小为4096),通过页号找到块号,通过块号找到块的起始地址,再加上页内偏移就可找到数据的物理内存地址
• 虚拟地址构成:页号 + 页内偏移 eg:OX12341234
• 页号:虚拟地址 / 快大小
• 页内偏移:虚拟地址 % 块大小
页内偏移
将物理内存中的一个块(大小为4096)放大,分析如下:
3.4 分段式
将物理地址分成一页一页的小块,分段式分的块大于分页式
• 虚拟地址构成:段号 + 页内偏移
3.5 段页式
分段式是先通过段号找到页表的起始位置,然后通过页表的起始位置找到页号,再通过页号找到块号,通过块号找到块的起始地址再加上页内偏移,即可找到物理内存地址
• 虚拟地址构成:段号 + 页号 + 页内偏移