电脑基础 · 2023年3月21日

<Linux>环境变量

环境变量

<Linux>环境变量

文章目录

  • 环境变量
    • 一、基本概念
    • 二、常见环境变量
    • 三、查看环境变量的方法
    • 四、测试PATH
    • 五、测试HOME
    • 六、测试SHELL
    • 七、环境变量相关的命令
    • 八、环境变量的组织方式
    • 九、命令行参数
    • 十、通过代码获得环境变量
    • 十一、通过系统调用获取环境变量
    • 十二、环境变量通常是具有全局属性的

一、基本概念

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

例如,我们编写的C/C++代码,在各个目标文件进行链接的时候,从来不知道我们所链接的动静态库在哪里,但是照样可以链接成功,生成可执行程序,原因就是有相关环境变量帮助编译器进行查找。

环境变量通常具有某些特殊用途,并且在系统当中通常具有全局特性。

我们都清楚自己写的一串代码,经过编译后生成可执行程序,我们用./即可运行,但是系统里有些命令也是64位的可执行程序:

<Linux>环境变量

  • 既然都是程序,那就可以把你自己的写的程序叫做指令,把系统的指令叫做命令程序or二进制文件。所以自己写的程序和系统中的指令没区别,均可以称为指令、工具、可执行程序。

二、常见环境变量

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

三、查看环境变量的方法

我们可以通过echo命令来查看环境变量,方式如下:

echo $NAME //NAME为待查看的环境变量名称

例如,查看环境变量PATH。

echo $PATH

<Linux>环境变量

四、测试PATH

大家有没有想过这样一个问题:为什么执行ls命令的时候不用带./就可以执行,而我们自己生成的可执行程序必须要在前面带上./才可以执行?
<Linux>环境变量
容易理解的是,要执行一个可执行程序(指令)必须要先找到这个程序,既然不带./就可以执行ls命令,说明系统能够通过ls名称默认帮助我们找到ls的位置,而系统是无法找到我们自己的可执行程序的,所以我们必须带上./,以此告诉系统该可执行程序位于当前目录下。

而系统就是通过环境变量PATH来找到ls命令的,查看环境变量PATH我们可以看到如下内容:
<Linux>环境变量
可以看到环境变量PATH当中有多条路径,这些路径由冒号隔开,当你使用ls命令时,系统就会查看环境变量PATH,然后默认从左到右依次在各个路径当中进行查找。
而ls命令实际就位于PATH当中的某一个路径下,所以就算ls命令不带路径执行,系统也是能够找到的。
<Linux>环境变量

那可不可以让我们自己的可执行程序也不用带路径就可以执行呢?

当然可以,下面给出两种方式:

方式一:将可执行程序拷贝到环境变量PATH的某一路径下
既然在未指定路径的情况下系统会根据环境变量PATH当中的路径进行查找,那我们就可以将我们的可执行程序拷贝到PATH的某一路径下,此后我们的可执行程序不带路径系统也可以找到了。

sudo cp myproc /usr/bin/

<Linux>环境变量

但是并不建议把你自己写的可执行程序随意添加到系统里,因为我们写的程序没有经过测试,可能会污染指令池。我们执行下面的命令删除即可:

sudo rm /usr/bin/myproc

方式二:使用exprot命令将可执行程序所在的目录添加到环境变量PATH中
将可执行程序所在的目录导入到环境变量PATH当中,这样一来,没有指定路径时系统就会来到该目录下进行查找了。

export PATH=$PATH:/home/wei/106/linux/lesson13

<Linux>环境变量
将可执行程序所在的目录添加到环境变量PATH当中,并且在后面追加myproc的当前路径。位于该目录下的可执行程序也就可以在不带路径的情况下执行了。PATH中不能由空格,因为Linux中以空格为分隔符。
<Linux>环境变量

如果想删除这个环境变量的话,执行【unset】命令。

我们通过下图也能发现:命令行可以定义本地变量和环境变量。

<Linux>环境变量

学过Java的同学应该都知道,我们在最开始学习Java时需要在Windows中配置环境变量,其实其本质就是向PATH中添加内容 (Windows中的路径以分号分割)

<Linux>环境变量

五、测试HOME

任何一个用户在运行系统登录时都有自己的主工作目录(家目录),环境变量HOME当中即保存的该用户的主工作目录。

普通用户示例:
<Linux>环境变量
超级用户示例:
<Linux>环境变量

六、测试SHELL

我们在Linux操作系统当中所敲的各种命令,实际上需要由命令行解释器进行解释,而在Linux当中有许多种命令行解释器(例如bash、sh),我们可以通过查看环境变量SHELL来知道自己当前所用的命令行解释器的种类。
<Linux>环境变量
而该命令行解释器实际上是系统当中的一条命令,当这个命令运行起来变成进程后就可以为我们进行命令行解释。
<Linux>环境变量

环境变量是怎么来的
我们 “ls -al /home/wei” 可以发现家目录下存在两个隐藏文件 – .bash_profile 与 .bashrc:

<Linux>环境变量

.bash_profile

<Linux>环境变量

.bashrc

实际上,当我们在登录 shell 时,操作系统会让我们当前的 shell 进程执行 .bash_profile 中的内容,而 .bash_profile 又会调用执行 .bashrc,它们会将对应的环境变量导入到 shell 进程的上下文环境中。所以,如果我们上面不小心将 $PATH 覆盖掉了也不用担心,重新登录 shell 就好了。

至此,环境变量的定义如下:

环境变量是操作系统为了满足不同的应用场景,预先在系统内设置的一大批全局变量,这些变量往往具有特殊功能,且能够一直被 bash 以及 bash 的子进程访问。

注:环境变量具有全局属性的根本原因是环境变量会被子进程继承。

七、环境变量相关的命令

1、echo:显示某个环境变量的值
<Linux>环境变量

2、export:设置一个新的环境变量

我们也使用export可以导出环境变量,使用env显示环境变量

<Linux>环境变量

3、env:显示所有的环境变量
<Linux>环境变量
部分环境变量说明:

环境变量名称 表示内容
PATH 命令的搜索路径
HOME 用户的主工作目录
SHELL 当前Shell
HOSTNAME 主机名
TERM 终端类型
HISTSIZE 记录历史命令的条数
SSH_TTY 当前终端文件
USER 当前用户
MAIL 邮箱
PWD 当前所处路径
LANG 编码格式
LOGNAME 登录用户名

4、set:显示本地定义的shell变量和环境变量

不加export定义的就是本地变量,可以通过set命令查看本地变量,也可以查看环境变量:

<Linux>环境变量

5、unset:清除环境变量
<Linux>环境变量

八、环境变量的组织方式

在系统当中,环境变量的组织方式如下:
<Linux>环境变量
每个程序都会收到一张环境变量表,环境表是一个字符指针数组,每个指针指向一个以’\0’结尾的环境字符串,最后一个字符指针为空。


九、命令行参数

前面获取环境变量是通过命令行的方式,下面我将通过代码程序来演示如何获取环境变量。

问:main函数可以带参数吗?最多可以带多少?

其实main函数的参数有3个,只是平时不经常用到,所以不太熟悉他们:

int main(int argc, char* argv[], char* envp[])
{
    return 0;
}

int argc:数组里的元素个数

char argv[ ]:指针数组*

我们先谈它的前两个参数,先来一段代码作为测试用例:

<Linux>环境变量

运行结果如下:

<Linux>环境变量

main函数的第二个参数是一个字符指针数组,此数组的下标0的位置存储的是你命令行的第一个位置(可执行程序),所以这里是第一行为:argv[0]:./myproc。其余的字符指针存储的是命令行对应的选项,所以会出现argv[1]:-a……,而main函数里的第一个参数存储的是数组里的元素个数。

<Linux>环境变量

总结:我们给main函数传递的argc,char* argv[ ]是命令行参数,传递的内容是命令行中输入的程序名和选项,并且结尾以NULL束!!!

**问:**main函数传这些参数的意义是什么?

下面我们可以尝试编写一个简单的代码,该代码运行起来后会根据你所给选项给出不同的提示语句。

#include <stdio.h>                                                                                           #include <string.h>
int main(int argc, char *argv[], char* envp[])
{
	if(argc > 1)
	{
		if(strcmp(argv[1], "-a") == 0)
		{
			 printf("you used -a option...\n");
		}
		else if(strcmp(argv[1], "-b") == 0)
		{
			printf("you used -b option...\n");
		}
		else
		{
			printf("you used unrecognizable option...\n");
		}
	}
	else
	{
		printf("you did not use any option...\n");
	}
	return 0;
}

<Linux>环境变量

**综上:**同一个程序,通过传递不同的参数,让同一个程序有不同的执行逻辑,执行结果。Linux系统中,会根据不同的选项,让不同的命令,可以有不同的表现,这就是指令中各个选项的由来和起作用的方式!!这也就是命令行参数的意义,同样也就是main函数参数的意义。

下面来谈下main函数的第三个参数:

int main(int argc, char* argv[], char* envp[])
{
    return 0;
}

char*envp[] 就是环境变量。也是一个字符指针数组,前面的argv是指向的命令行参数字符串,而这里envp指向的是一个一个环境变量字符串,最后也是以NULL结尾。结构图如下:

<Linux>环境变量

我们以如下代码来测试第三个参数:

int main(int argc, char* argv[], char* envp[])
{
	for (int i = 0; env[i]; i++)
	{
		printf("envp[%d]: %s\n", i, envp[i]);
	}
	return 0;
}

<Linux>环境变量

**总结:**一个进程是会被传入环境变量参数的。

**补充:**一个函数在声明和定义的时候都没有参数,那么我实际传参的时候可以传参。


十、通过代码获得环境变量

上面在学习main函数的三个参数的时候,我们得知通过main函数的第三个参数可以获得环境变量:

<Linux>环境变量

<Linux>环境变量

我们也可以通过第三方变量environ获取:

<Linux>环境变量

<Linux>环境变量


十一、通过系统调用获取环境变量

除了通过main函数的第三个参数和第三方变量environ来获取环境变量外,我们还可以通过系统调用getenv函数来获取环境变量。

getenv可以通过目标环境变量名进行查找,返回一个对应的字符指针,从而直接获得环境变量的内容:

<Linux>环境变量

<Linux>环境变量

问:我为什么要获得环境变量?

假设我们当前的用户USER为wei,但是我只允许自己使用,不允许其它人访问,就可以用获得环境变量来解决:

<Linux>环境变量

此时用wei用户是可以正常访问的,但是zxy用户就不能了:

<Linux>环境变量

综上,环境变量一定在某些地方有特殊用途,上面粗略的展示了其中一个方面。


十二、环境变量通常是具有全局属性的

先来回顾下bash进程:

<Linux>环境变量

我们先前说过,子进程pid每次运行的结果是在不断变化的,因为其每次运行,进程都在重启,但是父进程是不变的,因为父进程就是bash,是系统创建的命令行解释器,

<Linux>环境变量

如果我现在把bash杀掉,出现的结果是我输入任何命令都没有反应了,命令行直接挂掉。所以我们能够正常使用命令行,是因为这些命令本身就是被bash进程获得的。而这个bash进程是在我们登录系统时,系统自动帮你创建的。命令行中启动的进程,父进程全都是bash。

下面来理解环境变量具有全局属性:

看如下代码:(在原有的pid和ppid基础上添加了获取环境变量)

<Linux>环境变量

通过运行结果得知:我们的进程刚开始并不存在环境变量,但是若我们自己导出一个环境变量,此时就获取到了环境变量

<Linux>环境变量

总结:

环境变量是会被子进程继承下去的!!类比于从bash进程开始,往后创建一大批的子进程,若在bash进程的位置就创建了export环境变量,那么此环境变量就会从定义处bash位置开始被所有的子进程拿到。所以环境变量具有全局属性!!!

而本地变量之所以在本地有效,是因为本地变量是在bash内部定义的变量,不会被子进程继承下去!!!

补充:

<Linux>环境变量

  • 这里local_val是一个本地变量,命令行中启动的所有程序都要创建子进程,echo也是一个命令、一个子进程, 你父bash内定义的local_val变量怎么能被子进程读到呢?不是说本地变量不会被子进程读到吗?后面为何能用echo打出hello呢?

Linux下大部分命令是通过子进程的方式执行的!但是,还有一部分命令,不通过子进程的方式执行,而是由bash自己执行(调用己的对应的函数来完成特定的功能),我们把这种命令叫做内建命令。