[Linux]环境变量

一.什么是环境变量

为了满足不同的运行场景,操作系统预先设置了一大批全局变量,这种可以指定操作系统运行环境的变量就是环境变量。

我们平常使用的指令本质上也是用C语言实现的一个个小程序,但是我们在执行我们自己的可执行程序时往往是类似这样的:./test这个意思是告诉操作系统我要执行的是当前路径下的test文件(执行一个文件首先要找到这个文件)。

使用系统指令的时候不需要带./的原因就是环境变量PATH中存放系统的默认搜索路径,我们的可执行程序不能直接执行,是因为可执行程序的路径不在PATH中。

在这里插入图片描述
在这里插入图片描述

存放指令的路径是usr/bin

1.让我们自己的程序不用带路径

a.将我们的文件放入指令池

执行ls等程序时不需要带路径是因为环境变量PATH中存放了系统的默认搜索路径,那如果我不想让我的程序在执行是带路径就只要将我的程序放入到系统的默认搜索路径中即可。
在这里插入图片描述

这里要使用sudo权限,因为拷贝在Linux就等同于安装

不建议这样,因为我们写的代码没有经过测试,直接放到系统的指令池中会将其污染

使用sudo rm /usr/bin/mytest在系统指令路径下删掉我们自己的程序

在这里插入图片描述

b.增添环境变量PATH

可以在环境变量PATH中加入我们的当前路径,这样我们的当前路径也就变成了系统的默认搜索路径。既可以不污染指令池又可以让我们的程序不带路径。

细节管们已经发现了在环境变量PATH中,路径之间是以:间隔的,所以在添加新路径的时候要在旧路径后面写:+路径

在这里插入图片描述

echo PATH=路径的方法是错误的,会将PATH中原本存在的值给覆盖掉

在这里插入图片描述

此时你再使用ls命令,它会告诉你command not found。这是因为它不会默认的再去usr/bin路径下查找了。

如果你不慎这么操作了,也没关系你只需要关掉服务器再重新启动一次,这个PATH就又会被加载成最初的模样了。

在这里插入图片描述

etc/bashrc这个文件中存放的就是各种环境变量,每当我们启动一次,这个加载操作就会被执行。


二.环境变量相关的指令

1.set:显示本地的shell变量和环境变量

2.unset:取消环境变量

3.export:将本地变量设置成环境变量

所谓的本地变量就是我们直接在bash上定义的变量,这样的变量是本地变量只在当前进程(bash)有效,不可以被子进程继承而环境变量可以被子进程所继承

在这里插入图片描述

4.env:显示所有的环境变量

在这里插入图片描述

系统之所以能知道我当前的路径是因为有个环境变量叫PWD,系统之所以能判断我有没有权限是因为有个环境变量叫USER,sudo就是将USER变量改成了root所以我就有了root权限。

5.echo:显示某个环境变量的值(echo+$变量名甚至可以查到本地变量)


三.获取环境变量

如果你想在程序中显示某个环境变量,就可以使用getenv这个系统调用函数来获取

在这里插入图片描述

可以看到它的参数是字符指针,也就是说环境变量其实都是字符串

在这里插入图片描述

因为不同用户的环境变量不同,所以结果也不同。下面我切换到root用户再执行一次这个路径的这个程序,一起来看看结果

在这里插入图片描述

实现pwd指令

既然环境变量中有一个PWD,而我现在又有了拿到环境变量的getenv函数,所以实现pwd命令对我们来说简直是有手就行。

#include <stdio.h>
#include <stdlib.h>
#define MYPWD "PWD"
int main()
{
    char* p=getenv(MYPWD);
    printf("%s",p);
    return 0;

在这里插入图片描述

你还可以将这个命令放到默认搜索路径下,以后这个程序就是你的另一个pwd指令

四.命令行参数

main函数也是可以有参数的(最多可以有3个),接下来我们首先看一个现象

在这里插入图片描述

如果gcc版本比较低就要主动声明一下-std=c99,否则不能支持c99标准中的某些写法

好奇怪我们打印argv数组中的内容时拿到的确实我们的可执行程序名以及我们带的选项。

所以我们可以得到这样的一个结论:命令行输入的命令其实是一个大字符串,这个字符串会被拆开然后按顺序传给argv数组

在使用指令的时候,我们不但可以单独使用指令,还可以使用指令带选项的方式,就是因为命令行的前两个参数。

int main()
{
	if(argc != 2)
    {
        printf("Usage: \n\t%s [-a/-b/-c/-ab/-bc/-ac/-abc]\n", argv[0]);
        return 1;
    }
    if(strcmp("-a", argv[1]) == 0)
    {
        printf("功能a\n");
    }
    if(strcmp("-b", argv[1]) == 0)
    {
        printf("功能b\n");
    }
    if(strcmp("-c", argv[1]) == 0)
    {
        printf("功能c\n");
    }
    if(strcmp("-ab", argv[1]) == 0)
    {
        printf("功能ab\n");
    }
    if(strcmp("-bc", argv[1]) == 0)
    {
        printf("功能bc\n");
    }
    return 0;
}

指令的设计就是使用这种方式,在函数中对输入指令进行判断,再执行对应指令。无非就是使用C语言对具体功能进行封装。


接下来我再将第三个参数放进来:

在这里插入图片描述

在C语言中,指针类的参数一般都是以空指针作为结尾的。

可以看到直接就拿到了整个环境变量,这和在命令行输入env命令是一样的。

此外还可以通过environ拿到环境变量:

在这里插入图片描述

environ是系统中的一个全局的二级指针,它指向命令行参数表,作为参数传递给char* env[]。environ没有包含在任何头文件中,所以在使用时 要用extern声明。

在这里插入图片描述


我们平常也不用environ,但是程序也总能拿到环境变量是因为:环境变量本身是被加载到物理内存再映射到进程地址空间的,所以即使程序不去主动的获取环境变量,那些环境变量也被加载到虚拟内存中了。

五.总结

总之环境变量就是操作系统预先设在的一大批全局变量,当我们在启动计算机时这些全局变量就会加载到内存中可以指定计算机的运行环境。

  • 7
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 3
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

别动我的饭

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值