【linux进程(六)】环境变量再理解&程序地址空间初认识

简介: 【linux进程(六)】环境变量再理解&程序地址空间初认识

1. 前言

如果你不知道什么是环境变量PATH

请先阅读下面的文章:

初识环境变量

本章重点:

本篇文章着重于完善上一篇文章遗留
的一些环境变量的问题,认识其他几个
常见的环境变量,以及在bash下查看
环境变量和添加环境变量的方式,以及
拓展main函数的第三个参数:env.
最后对程序地址空间的内容开个头!


2. 在bash中查看所有环境变量

在bash中查看所有环境变量:

使用指令: env

可以看见打印出密密麻麻的环境变量

这其中不需要我们全部记住,只需简单

的认识几个常见的环境变量即可!

  1. 环境变量PWD:

记录当前路径,你以为平时执行的
pwd指令是怎么知道当前路径的?
稍加思考就能窥探,pwd实际上调用
了PWD环境变量!

  1. 环境变量HOME

我们平时使用的指令:cd ~进入
家目录,哪儿是家目录?环境变量
HOME里面的内容就是家目录!

  1. 简述登录Xshell时,所要做的事

根据以上对环境变量的认识,我们发现
登录xshell时我们用root账户登录就
会跑到root的家目录下,这是因为在
启动Linux时,OS帮我们做了一些事情

  • 输入用户名,密码
  • 认证信息是否正确
  • 形成环境变量(PATH,PWD,等等)
  • 根据用户名初始化HOME环境变量
    HOME=root或HOME=XXX
  • cd $HOME(进入家目录)

3. 程序中获取环境变量的方式

我们知道,Linux是由C语言写的

所以这里的程序我们默认是C程序

使用函数: getenv()

使用方法如下:

char* p = NULL;
p = getenv("PATH");

p就能获取环境变量PATH的内容了

即函数的参数是环境变量的名字

返回值是环境变量的内容!


4. main函数的第三个参数解析

不卖关子,直接写出来:

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

第三个参数为env,是不是很熟悉?
没错,第三个参数就是环境变量

它和第二个参数一样是指针数组

它指向环境变量表,环境变量表中的

内容和在bash中使用env打印的一样!

到目前为止了解到,系统在启动

程序时会给main函数提供两张表:

  • 命令行参数表
  • 环境变量表

通过下面的代码查看环境变量表:

#include<stdio.h>
int main(int argc,char* argv[],char* env[])
{
  for(int i=0;env[i]!=NULL;i++)
  {
    printf("[%d]: %s"\n,i,env[i]);
  }
  return 0;
}

5. 本地的与env中的环境变量

我们可以在bash中直接定义环境变量

使用指令: 环境变量名=内容

比如:

用户自己定义的环境变量是本地的

而env并不能查看本地的环境变量

所以使用env查找刚才定义的环境变量

时,实际上会找不到!

如果想要我们定义的环境变量
被放在系统的环境变量表中应该怎么做?

使用指令: export 环境变量名

还可以使用export定义环境变量:


6. 配置文件与环境变量的全局性

根据上面的一些信息我们可以推论:

在修改/定义环境变量时,实际上
修改/定义的是bash进程内部的
环境变量信息,每一次重新登录都会
形成新的bash解释器,并且新形成
的bash解释器会自动从家目录下
bash_profile文件中读取信息形成
一份新的环境变量表!

所以为啥每次重新启动bash后
我们自己定义/修改的环境变量
就不见了?因为它并没有被保存在
配置文件bash_profile中!

查看配置文件bash_profile:

当我们在配置文件中加上自己想要

一直存在的环境变量时,此时bash再

次启动后,这个环境变量就会被放到

环境变量表中!

并且在创建子进程时
这张表会通过main函数的参数传递
给子进程,所以子进程有了和父进程
一样的环境变量表,而子进程创建孙子
进程时又会将子进程的表给孙子进程
所以说,环境变量具有全局性!

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

7. 内建命令与常规命令

我们知道一个事实,bash中的指令

可以直接使用,不用加./是因为它的

路径在环境变量PATH中,所以我们

将PATH置空后,这些命令就不能运行了!

我们会发现一个问题,PATH被置空后

确实有些命令是跑不了了.但是某些

命令还是能跑,比如pwd命令还能正常

输出,这是为啥呢?

Linux下的命令父分类:

  • 常规命令:bash创建子进程执行的
  • 内建命令:bash自己执行的(类似于一个函数)

很明显pwd是内建命令,echo也是

内建命令,难道你就没有一个疑惑吗?

为什么我们自己定义的环境变量在
环境变量表env中找不到,但是echo
命令却可以将它的内容打印出来?
echo程序也是bash的一个进程
为什么它能获取我们定义的环境变量?
答案是echo是内建命令,是父进程
自己内部执行的,所以它可以看见
父进程的变化,但是其他的子进程不能
接受父进程的本地变量,所以看不见!


8. 程序地址空间初认识

C/C++程序员认为,程序的内存分布是这样的:

堆是向上增长的,栈是向下增长的

这个我们在以前的学习中以及有所

了解了,那为什么这里还要提呢?

有两个目的:

  1. 再对以前的知识做复习
  2. 在原有知识的基础上拓展

这其中有两个地方需要注意:

在写C程序时定义的常量字符串
实际上在上面区域的正文代码区
因为它和正文代码区紧挨着,所以
常量区的字符串不允许修改!
第二点是,栈虽然说是向下增长的
但是栈中的数组,结构体等结构的
地址是向上增长的,比如说开辟数组
时,开辟十个空间,那么数组中第一个
元素在空间的最下面,也就是地址最低
处,然后依次往上放后面的元素


9. 深入探究地址空间与物理内存

现在写上这样一段代码来深入探究地址:

#include <stdio.h>    
#include <unistd.h>    
#include <sys/types.h>    
#include<stdlib.h>    
int main()    
{    
    int id=fork();    
    int tmp=10;    
    if(id==0)//子进程执行的代码    
    {    
        tmp=20;    
        while(1)    
        {    
            printf("子进程,tmp: %d,&tmp: %p\n",tmp,&tmp);    
            sleep(1);    
        }    
    }    
    if(id>1)//父进程执行的代码    
    {    
        while(1)    
        {    
            printf("父进程,tmp: %d,&tmp: %p\n",tmp,&tmp);    
            sleep(1);    
        }    
    }                                                                                                                                                                   
    return 0;    
}

我们发现,tmp的值在父子进程中

是不同的,可以我们暂时可以理解,

是因为写时拷贝, 但是相同的地址
为什么会有不同的值?
这就有一点

挑战我们之前学习的知识了!

以目前的知识储备是无法想明白这个问题的

但是我们可以得出一个小结论:

我们平时写的堆区,栈区,静态区
等等区,并不是真实的物理地址!
因为一个物理地址不可能有两个值
那么真实的物理地址到底是什么?
如果这个不是真实的物理地址那么
这个地址又是什么东西呢?我们打印
出来的地址又是什么东西?以上内容
将在下一篇文章讲解!


10. 总结

了解环境变量相关知识可以更好的

理解操作系统在启动时所做的工作,

以及可以更好的了解进程在运行的

时候需要的一些数据从何而来!

本篇文章不再做拓展,更多内容
将在下一篇文章中讲解


🔎 下期预告:程序地址空间 🔍


相关文章
|
1月前
|
安全 Linux Shell
Linux上执行内存中的脚本和程序
【9月更文挑战第3天】在 Linux 系统中,可以通过多种方式执行内存中的脚本和程序:一是使用 `eval` 命令直接执行内存中的脚本内容;二是利用管道将脚本内容传递给 `bash` 解释器执行;三是将编译好的程序复制到 `/dev/shm` 并执行。这些方法虽便捷,但也需谨慎操作以避免安全风险。
|
2月前
|
网络协议 Linux
Linux查看端口监听情况,以及Linux查看某个端口对应的进程号和程序
Linux查看端口监听情况,以及Linux查看某个端口对应的进程号和程序
145 2
|
2月前
|
Linux Python
linux上根据运行程序的进程号,查看程序所在的绝对路径。linux查看进程启动的时间
linux上根据运行程序的进程号,查看程序所在的绝对路径。linux查看进程启动的时间
47 2
|
8天前
|
消息中间件 分布式计算 Java
Linux环境下 java程序提交spark任务到Yarn报错
Linux环境下 java程序提交spark任务到Yarn报错
18 5
|
22天前
|
Python
惊!Python进程间通信IPC,让你的程序秒变社交达人,信息畅通无阻
【9月更文挑战第13天】在编程的世界中,进程间通信(IPC)如同一场精彩的社交舞会,每个进程通过优雅的IPC机制交换信息,协同工作。本文将带你探索Python中的IPC奥秘,了解它是如何让程序实现无缝信息交流的。IPC如同隐形桥梁,连接各进程,使其跨越边界自由沟通。Python提供了多种IPC机制,如管道、队列、共享内存及套接字,适用于不同场景。通过一个简单的队列示例,我们将展示如何使用`multiprocessing.Queue`实现进程间通信,使程序如同社交达人般高效互动。掌握IPC,让你的程序在编程舞台上大放异彩。
15 3
|
2月前
|
Linux Windows Python
最新 Windows\Linux 后台运行程序注解
本文介绍了在Windows和Linux系统后台运行程序的方法,包括Linux系统中使用nohup命令和ps命令查看进程,以及Windows系统中通过编写bat文件和使用PowerShell启动隐藏窗口的程序,确保即使退出命令行界面程序也继续在后台运行。
|
5月前
|
存储 Shell Linux
【Shell 命令集合 系统设置 】⭐ Linux 取消或删除已设置的环境变量 unset命令 使用指南
【Shell 命令集合 系统设置 】⭐ Linux 取消或删除已设置的环境变量 unset命令 使用指南
198 0
|
2月前
|
Linux Shell
在Linux中,如何设置环境变量?
在Linux中,如何设置环境变量?
|
5月前
|
存储 Linux 编译器
Linux 交叉编译第三方库需要设置的环境变量
Linux 交叉编译第三方库需要设置的环境变量
175 0
|
5月前
|
Java Linux 开发工具
手把手教你Linux查找Java的安装目录并设置环境变量以及Linux下执行javac未找到命令的保姆级教学
手把手教你Linux查找Java的安装目录并设置环境变量以及Linux下执行javac未找到命令的保姆级教学
60 0
下一篇
无影云桌面