【Linux环境变量与命令行参数】常见环境变量 | 环境变量的全局属性 | 命令行参数

【写在前面】

对于环境变量,主要介绍基本概念及三四个环境变量 —— PATH、HOME、SHELL、HISTSIZE,其中 PATH 作为 “ 敲门砖 ”,我们会更详细讲解;理解环境变量的全局属性 —— 环境变量是可以被子进程继承(注意区分 C++ 里的继承);环境变量的组织方式。其次会介绍命令行参数 —— main 函数的参数。注意学习了本文的知识,并不代表己经掌握了环境变量,因为还有很多关于环境变量使用的 技巧及常见的环境变量没有过多的介绍,后面有需要用到环境变量的地方在展开。

一、环境变量

💦 基本概念

在 Linux 及 Windows 系统中,存在一种特殊的系统级变量,我们称之为环境变量。

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

    结合下面的理解,环境变量一般是在操作系统启动之后生成的,这些变量的数据来源,是通过一些某些配置文件来的。

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

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

💦 查看环境变量方法及相关命令

echo $NAME:NAME 是环境/本地变量的名称,查看 NAME 变量的内容。

env:查看系统中大部分的环境变量。

set:查看本地定义的变量和环境变量。

export:设置新的环境变量。

unset:清除设置的环境变量。

💦 常见环境变量
1. PATH,指定命令的搜索路径
  • echo $PATH可以看到 PATH 里的内容类似于路径。

    在这里插入图片描述

    当代码写完,经过编译,产生了一个可执行程序 mytest,那么 mytest 是一条命令吗 ❓

    在这里插入图片描述

    在这里插入图片描述

    在 Linux 中,你自己生成的可执行程序,具有可执行权限 x,所以可执行程序也是一条命令。从现在开始我们就可以认为程序、可执行程序、指令、命令等,都是一个概念。也就是说 ./mytest 和 ls 这样的系统级命令没有差别。

  • 大家都是程序,为什么系统级命令不用加 ./,而自己的命令却要加 ./ ❓

    在这里插入图片描述

    ./ 意思是当前路径。所以我们就能理解:ls 是系统级命令,路径不在当前,所以 ./ls 就会 No such file or directory;mytest 是自己的命令,且在当前路径,所以 ./mytest 可以执行;这里不理解的是 ls 是怎么找到的,以及 mytest 为啥找不到。

    其中 mytest 报的是 -bash: mytest: command not found ,那么就意味着曾经系统有找过,没找到,那么系统在哪里找 ? ? ?

    系统在进行查找命令时,默认是在环境变量 —— PATH 中查找的,而 PATH 在这里的功能是辅助系统进行指令查找,这是 PATH 存在的价值。所以这就解释了为啥 ls 去 PATH 里找得到,mytest 去 PATH 里找不到,./mytest 在当前路径找得到,./ls 在当前路径找不到的原因。

    也就是说 PATH 里保存的就是当前指令的搜索路径,我们再来了解下PATH 的路径 ???

    在这里插入图片描述

    其中冒号是每个路径之间的间隔符;而每个路径是绝对路径。也就是说系统在执行 ls 、 mytest 等指令时默认会一个路径一个路径的查找,如果找到了就执行,如果到最后也没找到,那么就会 -bash: xxx: command not found。

  • 实现 mytest (这里有两种方法)

    1. 把 mytest 拷贝至 PATH 下的任何一个路径下就可以实现 mytest

      在这里插入图片描述

      当把上一步中 cp 的删除再运行和没有 cp 过的运行对比,它们报的错误不一样,并且它还会去 /usr/bin/ 下查找,原因是 linux 系统把前面 cp 的路径缓存了

    2. 把 mytest 当前所处路径添加到 PATH 下

      PATH=$PATH:/home/DanceBit/,其中新的路径 = 原来的路径 : 要添加的路径。注意如果没有 $PATH,那么老的路径会被全部替换成要添加的路径。

      在这里插入图片描述

      如果 PATH=/,就意味着会把 PATH 给清空了,那么原本系统级的命令 ls 等就使用不了了,pwd 能使用的原因是因为 pwd 不在 PATH 下

      在这里插入图片描述

      那么也不要慌,PATH 既然是变量,意味着它是可以被赋值的 (上面就二次赋值过),那么最开始的一串的默认路径肯定是 Linux 系统在给你这个用户在配置文件里加载下来的。所以我们直接重新登录即可恢复默认的路径。

      在这里插入图片描述

      通过这里就可以知道,为啥很不不建议小白乱修改 Linux 系统中任何一个配置文件。没有改配置文件还好,大不了重新登录;但是改了配置文件,那么很多未知的错误就会出现。

  • 也就是说之前把要添加的路径配置到 PATH 下时,在下一次登录时就不复存在了,如何让它永久存在呢 (这里有两种方法)

    1. 修改相关配置文件,但是对小白来说极其不推荐。

      当系统登录成功后,系统会把各种需要的脚本跑一下,如 .bash_profile 和 .bashrc,然后我们就看到 echo $PATH 里的内容了。

      在这里插入图片描述

    2. 把程序拷贝到 PATH 下的任何一个路径下 (上面已演示)

      在这里插入图片描述

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

为什么登录不同的用户 DanceBit 和 root 时,所对应的家目录不一样 ❓

在这里插入图片描述

根本原因是因为 Linux 系统中还有一个环境变量 —— HOME,它默认表明当前用户登录时所处的默认路径,当然它在系统中也有配置的。我们从上图可以看到 DanceBit 用户中 HOME 保存的是 /home/DanceBit;而 root 用户中 HOME 保存的是 /root。

我们知道 Linux 系统是可以多人同时登录使用的,其实 Windows 也是可以多人登录使用的,但是你和你的室友同时使用你室友的电脑有什么意义呢,你不也有吗 !!!

在这里插入图片描述

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

我们的命令行解释器有bashsh 等,可以看到 SHELL 里保存的是当前命令行解释器的种类 —— bash。

在这里插入图片描述

4. HISTSIZE

history记录了当前用户历史上所使用的部分命令

在这里插入图片描述

history | wc -l统计了历史命令有 15 条 (加上自己的一条)

在这里插入图片描述

echo $HISTSIZE可以看到里面保存着3000,你按键盘的上和下能翻到历史命令,前提条件是系统给你保存了,但是如果保存太多,对系统也是负担,所以默认最多只能保存 3000 条。

在这里插入图片描述

💦 环境变量的全局属性
  • 注意环境变量的全局属性和语言上的全局属性不一样,比如说你定义了一个全局变量,这个全局变量就可以被当前文件中所有函数访问,而环境变量的全局属性指的是它可以被子进程继承下去。

    这里我们学习两个获取环境变量的函数 —— getenv(主文主要学习)、setenv,它们都是库函数。

    在这里插入图片描述

  • 其中使用 getenv 的参数是环境变量的名称,返回值是该变量对应的内容

    在这里插入图片描述

    这里可以看到运行后它可以把环境变量的内容输出,是因为子进程继承了父进程 bash 的环境变量。在命令行上运行的 大部分 指令,它的父进程都是 bash,不管是你的程序,还是系统的命令,都是 bash 创建子进程,然后子进程执行你的命令,其中创建子进程我们知道了 fork,但子进程执行你的命令会在进程控制中会讲解。

  • 命令行中,我们可以定义两种变量 ❓

    1. 本地变量。

      这里MY_VAL是在当前 bash 内定义的变量,只能在当前 shell 命令行解释器内被访问,不可以被子进程继承。

      在这里插入图片描述

      验证不可以被子进程继承,这里 ./mytest 运行后就是 bash 的子进程,显然本地变量并没有被子进程继承,且会报段错误。

      在这里插入图片描述

      export ???

      其中 export MY_VAL 就可以将本地变量 MY_VAL 转换成环境变量,而环境变量就可以被 bash 的子进程 mytest 所继承。

      在这里插入图片描述

      env 可以查看所有的环境变量,但如何查看所有本地变量 ???

      env用于查看所有环境变量;set用于查看所有本地变量和环境变量。

      在这里插入图片描述

      在这里插入图片描述

      unset ???

      用清除设置的环境变量。

      在这里插入图片描述

      echo 本地变量 ???

      在这里插入图片描述

      上面不是说命令行上运行的大部分指令,不管是你的程序,还是系统的命令,它的父进程都是 bash,且本地变量是不能被子进程继承的吗。echo 当然是命令,当 echo $val101 时也需要 bash 创建子进程 (实际是下面的解释),而 val101 作为本地变量却好像也能被继承的原因是像 echo, export, set, env 等命令,我们一般称之为 内建命令,可以理解为 shell 程序内部的一个函数,也就是说 shell 在执行命令时,如果是内建命令,那么它直接调用内建命令对应的方法,如果不是内建命令,那么就会 fork 子进程。要注意细节,我们上面说的是 大部分 指令,显然就不包括 echo 等小部分指令。

    2. 环境变量。

      环境变量具有 “ 全局属性 ”,它可以被子进程继承。

二、 main 函数的参数

💦 命令行参数
  • main 函数有没有参数,可不可以带参,如果可以最多可以带几个参数 ❓

    在这里插入图片描述

    咦!好奇怪。在 C/C++ 中写了这么多 main 函数从来没有写过带参的。实际 main 函数是可以带参的,可以带 3 个参数。其中int argc是命令行参数的个数;char* argv[]是指针数组,数组里有几个有效元素是由argc确定,所以 main 函数的前两个参数用来记录的是我们在命令行上传入的参数,我们称这两个参数为命令行参数

  • 为什么要存在命令行参数 ❓

    在 Windows 下是不好感受不到的。比如这里我们要实现加减法计算器./cal -add 3 5./cal -sub 3 5,那么它有4argc

    在这里插入图片描述

    有人说这也体现不了命令行参数有啥价值呀。可以看到下图,ls 命令可以根据不同的选项完成不同的功能,这就是命令行参数最大的价值。通过命令行参数就可以帮我们在同一个程序中设计出不同的业务功能。所以为什么我们在大部分 C/C++ 代码中不用命令行参数的原因是你的 main 函数只有一种功能。

    在这里插入图片描述

    比如 Windows 下shutdown命令,而因为命令行参数的作用,所以它能根据不同选项完成不同功能。

    在这里插入图片描述

    在 C语言里,我们学习函数栈帧时,说过 vs 环境下 main() 是被 __tmainCRTStartup() 调用的,__tmainCRTStartup() 又被 mainCRTStartup() 调用的 … …,其中 main 函数的参数是在 __tmainCRTStartup() 调用 main 函数时就传入了,Linux 下也是如此。

💦 char* env[]

char* env[]是 main 函数的第 3 个参数,它和第 2 个参数的类型一样都是指针数组。但是 env 的每个元素是环境变量。换言之,我们把环境变量的路径作为字符串,用 char* env[] 这样的字符指针数组,依次指向不同的环境变量,我们就可以通过数组传参的方式,把环境变量传递给当前程序,当前程序运行后成为进程,也就意味着进程拿到了环境变量。可以看到这里 ./test 打印出来的环境变量几乎是和 bash 的环境变量一模一样,说明环境变量的继承是通过第三个参数 char* env[] 来继承的。

在这里插入图片描述

在这里插入图片描述

💦 通过代码如何获取环境变量
  1. 命令行第三个参数。

    只要遍历 env 这个指针数组即可,只不过涉及文本处理,有点麻烦。

    在这里插入图片描述

  2. getenv

    现在我们就明白了getenv("PATH")的原理其实就是在 env 里,进行文本匹配,找到 PATH 对应的内容返回。

  3. 通过第三方变量environ获取

    在这里插入图片描述

    这个用法很少,了解一下即可。在 C 中想获取环境变量,可以使用 C 提供的第三方全局变量 —— environ。libc中定义的全局变量 environ 指向环境变量表,environ 没有包含在任何头文件中,所以在使用时要用extern声明。声明时是二级指针的原因是要接收字符指针数组,而数组名是一级指针。

    在这里插入图片描述

    在这里插入图片描述

    帮助理解 ???

    在这里插入图片描述

    咦!这有点接受不了了,难道学了个假的 C 语言吗,其实是还有些冷门的知识没学。这里实际 10 和 20 确实是传给了 show,只不过你没有办法使用,其中如果你曾经了解过函数调用时会形成栈帧,那么就知道 10 和 20 会被压入 show 这个栈帧中,换句话说,我们在 show 里是可以使用某些指针操作得到 10 和 20 的。同理 main 函数虽然没有明确写带参,但同样也可以把 argc、argv、env 传参,同样这里的二级指针也可以通过某种方式,在压栈结构中,让 environ 指向传入的命令行参数。

【写在后面】

上面的概念都掌握后并不代表已经完全掌握了环境变量,因为有很多关于环境变量的使用技巧和常见的环境变量并没有过多的介绍,后面的学习中有使用到环境变量时再谈。

评论 33
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

跳动的bit

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

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

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

打赏作者

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

抵扣说明:

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

余额充值