目录
1.基本变量
2.通过代码获取环境变量
2.1 main传参
2.2 全局变量environ
2.3 系统调用getenv()
3.在脚本文件中添加环境变量
4.环境变量通常是具有全局属性
1.基本变量
环境变量(environment variables)一般是指在操作系统中用来指定操作系统运行环境的一些参数如:我们在编写C/C++代码的时候,在链接的时候,从来不知道我们的所链接的动态静态库在哪里,但是照样可以链接成功,生成可执行程序,原因就是有相关环境变量帮助编译器进行查找。环境变量通常具有某些特殊用途,还有在系统当中通常具有全局特性。
常见环境变量
- PATH : 指定命令的搜索路径
- HOME : 指定用户的主工作目录(即用户登陆到Linux系统中时,默认的目录)
- SHELL : 当前Shell,它的值通常是/bin/bash。
- USER:当前帐号使用者
查看环境变量方法
echo $NAME //NAME:你的环境变量名称
env 查看所有环境变量
测试PATH
- 创建myproc.c文件
#include <stdio.h> int main() {printf("hello world!\n");return 0; }
- 对比 ./myproc 执行和之间 myproc 执行
- 要执行一个命令,必须先要找到这个对应的可执行程序!为什么有些指令可以直接执行,不需要带路径,而我们的二进制程序需要带路径才能执行?
- 将我们的程序所在路径加入环境变量PATH当中, export PATH=$PATH:myproc 程序所在路径
- 对比测试
- 还有什么方法可以不用带路径,直接就可以运行呢?那就是把我们写的可执行程序移动到PATH默认路径中。
默认更改环境变量,只限于本次登陆,重新登陆,环境变量会自动被恢复。那这是为什么呢?
因为这些环境变量的值是存在配置文件中的,每次登录都会重新读取配置文件。
测试HOME
- 用root和普通用户,分别执行 echo $HOME ,对比差异. 执行 cd ~; pwd ,对应 ~ 和 HOME 的关系
可以发现不同用户有不同的主目录即HOME。
为什么普通用户默认所处目录/home/xxxx,而root是 /root ???
登陆的时候
- 输入用户名&&密码
- 认证
- 形成环境变量(肯定不止一个,PATH,PWD,HOME)
- 根据用户名,初始化HOME。root:HOME=/root,普通用户:HOME=/home/xxxx
- cd $HOME
系统中会存在大量的环境变量,每一个环境变量都有它自己的特殊用途,用来完成特定的系统功能!
2.通过代码获取环境变量
获取环境变量有三种方式:
2.1 main传参
系统启动我们的程序的时候,可以选择给我们的进程(main)提供两张表
- 命令行参数表
- 环境变量表
命令行参数表之前文章已经讲过了。这里讲一下环境变量表。 我们执行下面代码,使用main函数的第三个参数
#include <stdio.h>
int main(int argc, char *argv[], char *env[])
{int i = 0;for(; env[i]; i++){printf("%s\n", env[i]);}return 0;
}
运行结果:
2.2 全局变量environ
libc中定义的全局变量environ指向环境变量表,environ没有包含在任何头文件中,所以在使用时 要用extern声明。不声明也可以。使用通过第三方变量environ获取环境变量:
#include <stdio.h>
int main(int argc, char *argv[])
{extern char **environ;int i = 0;for(; environ[i]; i++){printf("%s\n", environ[i]);}return 0;
}
2.3 系统调用getenv()
通过系统调用获取或设置环境变量:getenv()函数来访问特定的环境变量
#include <stdio.h>
#include <stdlib.h>
int main()
{//打印PATH的内容printf("%s\n", getenv("PATH"));return 0;
}
3.在脚本文件中添加环境变量
环境变量生成原理 :
命令行启动的进程都是shell/bash的子进程,子进程的命令行参数和环境变量是父进程bash给我们传递的!而父进程的环境变量信息是以脚本配置文件的形式存在的!
而我们直接在命令行更改的是bash进程内部的环境变量信息!每一次重新登陆,都会给我们形成新的bash解释器,并且新的bash解释器自动从读取形成自己的环境变量表信息!
每一次登陆的时候,你的bash进程都会读取 .bash_profile。配置文件中的内容,为我们bash进程形成一张环境变量表信息!
如果想要想要自己创建的一个环境变量在每次登录时都存在,可以在主目录下使用 ll -a,查看隐藏的 .bash_profile ,并对其进行修改
添加自己创建的环境变量:
查看是否成功:
4.环境变量通常是具有全局属性
- 环境变量通常具有全局属性,可以被子进程继承下去
我们在命令行中创建一个本地定义变量OURENV ,运行下面代码
#include <stdio.h>
#include <stdlib.h>
int main()
{printf("OURENV:%s\n",getenv("OURENV"));return 0;
}
运行结果:
可以发现:通过export 将局部变量添加到环境变量中,才能被命令行bash的子进程myproc得到。
本地变量vs环境变量
- 本地变量只在bash进程内部有效,不会被子进程继承下去
- 环境变量通过让所有的子进程继承的方式,实现自身的全局性!
如果我们把PATH该为空字符串,我们可以发现有些指令可以运行,而有些指令却不可以运行
Linux的命令分类:
- 常规命令,shell 通过fork让子进程成执行的,像ls touch ,所以PATH为空后不能执行。
- 内建命令,shell命令行的一个函数,向pwd,cd,是可以直接读取shell内部定义的本地变量
和环境变量相关的命令
- echo: 显示某个环境变量值
- export: 设置一个新的环境变量
- env: 显示所有环境变量
- unset: 清除环境变量
- set: 显示本地定义的shell变量和环境变量
没有export 的变量都是本地变量。可以通过set显示,unset清除
本篇结束!