【Linux】环境变量--PATH环境变量/环境变量的操作/命令行参数

一、PATH环境变量

1.什么是PATH环境变量

这里我们先提出一个问题:为什么我们运行自己写的程序需要带路径,而系统的指令不需要带路径?

这是因为系统能找到它的位置,系统中是存在相关的环境变量,保存了程序的搜索路径的,系统中搜索可执行程序的环境变量叫做PATH。所以我们只需要把我们的程序拷贝到/usr/bin/路径下即可(但是我们不推荐,理由下面解释)

首先我们理解什么是环境变量

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

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

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

所以事实上,Linux中的各种指令本质上是/usr/bin目录下的一个个可执行的程序,和我们自己编写的可执行程序没有任何区别。但是使用指令的时候,系统会自动去PATH中寻找指令,如果找到了就执行,没有找到就报错-“command not found”,所以Linux中的各种指令不需要指定路径,而我们自己写的程序需要指定路径

2.如何添加PATH环境变量

我们可以使用如下指令来查看PATH环境变量中包含的内容:

echo $PATH

在这里插入图片描述

我们需要注意的是,PATH中不同的路径分隔符为:

我们也可以通过向PATH中添加内容让我们自己的程序可以不指定路径就能被执行

方法一:直接将程序添加到/usr/bin目录下(不建议使用这种方法,因为我们写的程序没有经过测试,可能会污染指令池)

在这里插入图片描述

我们删除使用如下指令:

sudo rm /usr/bin/mycmd

之后我们执行我们自己写的程序又需要带上路径

在这里插入图片描述

方法二:使用export命令将当前可执行程序的路径导入PATH

export PATH=$PATH:当前路径

在这里插入图片描述

【注意】

$PATH代表之前的PATH中的内容,:之后的尾新添加的内容,所以我们不能直接使用如下指令:

export PATH=当前路径

因为这样会将之前PATH中的内容覆盖掉,使得Linux中的各种指令必须指定目录才能执行,因为操作系在/usr/bin找不到系统中原来的指令了,此时,我们只需要重新登录即可,因为环境变量属于内存级变量,我们每次登录shell的时候,环境变量都会重新编译执行。同时,PATH中不能有空格,因为Linux中以空格作为分隔符

在这里插入图片描述

我们学过Java的同学都知道,我们在最开始学的时候需要在windows中配置环境变量,其实本质就是向PATH中添加内容(windows中的路径分隔符为分号)

windows下查看环境变量:

设置->关于->高级系统设置->环境变量

在这里插入图片描述

3.系统中的其他环境变量

PATH环境变量只是系统中众多环境变量的一种,除了PATH,我们还有许多其他环境变量,不同的环境变量有不同的功能,也适用于不同的场景

PATH : 指定命令的搜索路径

HOME : 指定用户的主工作目录(即用户登陆到Linux系统中时,默认的目录)

SHELL : 当前Shell,它的值通常是/bin/bash

HOSTNAME:主机名

USER:当前用户名

PWD:当前系统路径

HISTSIZE:shell能记住的最多的历史命令的数量

在这里插入图片描述

我们也可以使用env命令来查看系统中所有的环境变量:

在这里插入图片描述

4.环境变量的来源

我们使用“ ls - al /hone/hdp”指令就可以发现家目录下 存在两个隐藏文件–.bash_profile与.bashrc:

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

实际上,我们在登录shell的时候,操作系统会让我们当前的shell进行执行.bash_profile文件内的内容,而.bash_profile又会调用执行.bashrc,他们会将对应的环境变量导入到shell进程的上下文环境中,所以,这样解释了为什么我们把$PATH覆盖了之后,重新登录shell之后就可以了

环境变量的定义

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

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

二、环境变量的操作

1.设置环境变量

Linux命令行的可以定义变量的,但是我们以这种方式定义的变量吧本地变量,即只在bash进程中有效,而不是环境变量,因为环境变量具有全局属性

在这里插入图片描述

我们可以使用export直接定义环境变量,也可以使用它将已经存在的本地变量变成环境变量

在这里插入图片描述

我们可以使用set命令来查看所有变量,包括环境变量和本地变量,使用unset来取消变量,包括环境变量和本地变量

在这里插入图片描述

2.通过getenv获取环境变量

我们可以使用 echo $环境变量名 来获取特定的环境变量,也可以通过getenv()函数来获取环境变量:

在这里插入图片描述

其中name是我们需要获取环境变量的名称,如果获取成功就返回环境变量的具体内容,失败就返回NULL

这样我们就可以使用getenv()函数来编写系统中某些指令了,比如pwd:

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

在这里插入图片描述

3.环境变量的意义

我们知道,环境变量是操作系统为了满足不同的应用场景,预先在系统内设置的一大批全局变量,其中PATH就是为了满足我们指令路径搜索的需求,而除了指令需求之外还有许多其他需求,其中非常重要的一个就是身份认证

我们以一个例子说明:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>

#define USER "USER"

int main()
{
    
    
    char* who = getenv(USER);
    if(strcmp(who,"root")==0)
    {
    
    
        printf("user:%s\n",who);
        printf("user:%s\n",who);
        printf("user:%s\n",who);
        printf("user:%s\n",who);
        printf("user:%s\n",who);
    }
    else
    {
    
    
        printf("权限不足\n");
    }
    
    return 0;
}

在这里插入图片描述

我们这里使用su -,而不是su,因为su -会重新登录shell,此时shell会重新加载环境变量,让$USER从hdp变成root,而su只是把用户身份切换为root

在这里插入图片描述

我们可以在程序内部通过getenv函数来获取当前Linux用户,然后判断其是否具备某种权限,再执行对应的操作

此外,我们还可以使用stat来获取一个文件的其他属性,比如读写执行等,然后根据这些属性判断一个用户是否能对该文件进行对应的操作

在这里插入图片描述

stat指令是一个用于显示文件或目录详细信息的命令,包括文件大小、创建时间、修改时间、访问时间和权限等。在Linux和Unix操作系统中,stat命令通常使用以下语法:

stat [选项] 文件名

其中,选项可以是以下之一:

  • -c:指定自定义格式输出。
  • -f:指定文件系统信息输出格式。
  • -t:指定时间格式输出。

三、命令行参数

我们出来可以通过getenv函数来获取环境变量,还可以通过命名行参数来获取环境变量

我们知道,C语言中的main函数是有参数的,这些参数可以通过命令行进行传递:

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

其中argc是一个整数,代表的是argv数组元素的个数,argv是一个指针数组,数组里面的每一个元素都指向一个字符串

我们可以打印argv数组中存放的环境变量,增加选项时环境变量也随之增加

#include <stdio.h>
#include <stdlib.h>

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

在这里插入图片描述

他们配合使用就可以实现类似于"ls -a -l -d"选项的功能,通过我们传递不同的选项使用不同的功能

#include <stdio.h>
#include <string.h>
#include <stdlib.h>

int main(int argc,char* argv[],char* env[])
{
    
    
    if(argc != 2)
    {
    
    
        printf("Usage: \n\t\%s [-a/-b/-c/-ab/-ac/-bc]\n",argv[0]);
    }
    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("-ac",argv[1]) == 0)
    {
    
    
        printf("功能ac\n");
    }
    if(strcmp("-bc",argv[1]) == 0)
    {
    
    
        printf("功能bc\n");
    }
    
    return 0;
}

在这里插入图片描述

而指针数组env就是用于接受父进程传递过来的环境变量的参数,我们可以在main函数中打印env的内容:

#include <stdio.h>
#include <stdlib.h>

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

在这里插入图片描述

程序也可以通过环境表environ来获取环境变量–环境变量表是一个字符指针数组,每个指针指向一个以’\0’结尾的环境变量字符串,每个进程都会收到一张环境变量表

在这里插入图片描述

在这里插入图片描述

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>

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

在这里插入图片描述

猜你喜欢

转载自blog.csdn.net/qq_67582098/article/details/134435620