【Linux仓库】命令行参数与环境变量【进程·伍】

举报
egoist2023 发表于 2025/09/15 18:30:36 2025/09/15
【摘要】 本文介绍了Linux系统中命令行参数和环境变量的相关知识。命令行参数通过main函数的argc和argv传递,允许程序根据用户输入执行不同操作,提升灵活性(如ls指令的选项实现)。环境变量是系统级全局变量,如PATH指定命令搜索路径,PWD记录当前目录。获取环境变量的方法包括main函数参数、getenv函数和environ变量。文章还解释了进程如何继承父进程环境变量,区分了环境变量和本地变量,并

目录

命令行参数

环境变量

命令行参数
main函数可以有参数吗?可以有几个参数呢?参数又是什么呢?带着这三个疑问往下阅读。

int main(int argc,char *argv[])
{
printf(“hello linux!\n”);
return 0;
}
我们或多或少在别人的程序中见过这种代码,清楚main函数是可以带参数的,并且可以有两个参数(实际上可以有3个参数,第3个是环境变量表)。那么这两个参数各代表什么含义呢?

argv是一个指针数组,它指向一个一个的字符串,最后以NULL结尾;
argc用来记录argv[]的元素个数。
我们的命令行参数会放到命令行参数表上,即argv,它用来存储命令行参数的每个单独的字符串。

意义
为什么要有命令行参数表呢?它能带来的意义又有哪些呢?

命令行参数传递:借助主函数的参数,能把命令行中用户输入的参数传递到程序内部,这样程序就能依据不同参数执行不同操作。
程序灵活性提升:无需对代码进行修改,通过命令行参数就能改变程序的运行方式或者配置信息。
从这两方面我们并不能体会到啊?因此小编写了段程序解释命令行参数带来的意义:

下图中写了一段程序,它借助main函数参数,让命令行参数传递的时候,通过匹配不同的命令行参数,如果第二个参数带的是v1则实现v1版本,带的是v2则实现v2版本,从而实现不同版本的逻辑,这带来的意义就是命令行参数传递实现不同的逻辑,同时我们也可以限制如果传递的命令行参数不符合我们的逻辑,则不能通过,提升了程序灵活性。

💻如图实现:

有没有一种可能我们之前所说的选项就是这样实现的呢?

是的没有错,Linux中许多指令也是借助了命令行参数才得以支持多模块实现,如ls指令。

ls -l -a -i
 ls的命令本质是一个程序,而我们之前说ls后面的是选项,今天我们对选项能有更深的理解,它是通过命令行参数匹配从而实现我们的目的。

-l 、 -a 本质是字符串,以一定的方式传递给ls内部的"main",在ls内部实现的时候,就可以根据不同的选项,实现类似功能的不同表现形式。

环境变量
引出概念:

环境变量(environment variables)⼀般是指在操作系统中⽤来指定操作系统运⾏环境的⼀些参数。如:我们在编写C/C++代码的时候,在链接的时候,从来不知道我们的所链接的动态静态库在哪⾥,但是照样可以链接成功,⽣成可执⾏程序,原因就是有相关环境变量帮助编译器进行查找。
环境变量是系统级别的一些全局变量,具备不同的用途(这也是为什么要有环境变量的原因)。
罗列Linux环境变量

PATH : 指定命令的搜索路径
HOME : 指定用户的主⼯作⽬录(即⽤⼾登陆到Linux系统中时,默认的⽬录)
SHELL : 当前Shell,它的值通常是/bin/bash。
PATH
我们前面说过,指令也是一个可执行程序。为什么有些指令可以直接执⾏,不需要带路径,⽽我们自己的⼆进制程序需要带路径才能执行啊( 否则会报找不到 )这是为什么呢?

通过which命令查看ls所处的路径,可以看到是在/usr/bin的路径下。 什么意思呢 ? 即OS要执行该指令时,就需要/usr/bin路径,那么由谁给它提供呢 ? OS默认会在PATH路径下查找,若没有则会报错。而/usr/bin被包含在PATH环境变量中。

因此我只要证明 /usr/bin 不在PATH中时,此时再使用 ls 指令会向我们执行自己的程序一样报没有找到 ;
或者将我们自己的程序添加到PATH中,再执行自己的程序,若不会报错, 即可证明OS是在PATH中查找的。

💻扩展:

OLDPWD

💻证明: echo $OLDPWD可以展示最近一次所处路径.

PWD
这里总算可以解释pwd指令的原理了:

我们知道每个进程(包括xshell啊!)都会记录自己所处的工作路径,这在之前是已经验证过的.那么我们的PWD环境变量又从哪里而来呢?

💻pwd指令原理:

多方法获取环境变量
获取环境变量的方法有很多,这里主要介绍三种:

main函数获取
我们前面说了main函数是可以带3个参数的,而第3个参数则是环境变量表.下图中

💻模拟实现env指令:

getenv获取

environ获取
libc中定义的全局变量environ指向环境变量表,environ没有包含在任何头⽂件中,所以在使⽤时 要⽤
extern声明。(实际上environ就是个二级指针,指向环境变量表)

我的进程如何获取环境变量 
我们大部分的指令以及自己实现的可执行程序,都不是有bash帮我们执行的,而是bash通过创建子进程的方式,让子进程继承了bash的环境变量,形成环境变量表。

环境变量全局属性
echo:显示某个环境变量值
export: 设置⼀个新的环境变量
unset: 清除环境变量
set: 显⽰本地定义的shell变量和环境变量

💻下图中证明本地变量是不可以被子进程所继承的:

MYENV是本地变量,执行程序时bash创建子进程,子进程以父进程为副本进行继承,包括环境变量.假设本地2变量可以被继承下去,那么getenv获取该变量,如果打印该变量不存在说明本地变量是不能被继承下去的.

内建命令
我们知道命令是可执行程序,其中的大部分命令需要通过创建子进程的性质执行。

那么肯定也存在一些命令是不需要通过创建子进程来执行,而是由bash来执行.那么这个命令在执行的时候是没有风险的.我们把这种命令叫做内建命令!!!

我们说本地变量是不能被子进程继承下来的,说 明本地变量只在bash自 身有效,如果能打印出该变量的值,恰恰说明echo指令是一个内建命令。

【版权声明】本文为华为云社区用户原创内容,未经允许不得转载,如需转载请自行联系原作者进行授权。如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。