近来小姐姐又犯憨憨错误,问组内小伙伴export
命令不会持久化环境变量吗?反正我是问出口了。。然后小伙伴就甩给了我一个《The Linux Command Line》PDF链接。感谢老大不杀之恩~
Shell是命令解释器,它会接受用户输入的各种命令,并传递给操作系统执行。它的作用类似于Windows系统的命令行。在UNIX或Linux系统中,Shell即是用户交互的界面,也是控制系统的脚本语言。当然现在用户也可以选择图形化界面做一些和操作系统的交互。层次示意图如下:
对于初学者来说,可能搞不清楚Shell怎么会有那么多分类,Shell的语法怎么那么随便...
小姐姐结合自己初学Shell傻傻分不清的问题点,主要从Shell的种类
,变量的分类
,条件测试的表达
三个部分来介绍。
Shell的种类
shell程序有sh,bash,zsh等分类,我从网上找到一张图可以看出shell程序的发展史。
对于这些Shell程序,其语法或多或少有一些差异,不过我们通常使用的都是bash。
- Shell程序信息
在Linux系统我们可以通过一些命令查看或修改当前Shell程序信息。
一般发行版的Linux系统中,默认的shell程序就是bash。我们在写shell脚本时,通常也会在脚本文件头部指定bash作为脚本解释器。
这里多说一句,zsh有时也作为猿媛们的默认shell。zsh语法大多是和bash匹配的,也不会影响shell脚本的执行(因为脚本头部指定bash就还是bash:),也不会影响像小姐姐这样的渣渣使用。用它是因为它有神奇的开源框架 Oh My God.. 哦不,是 Oh My Zsh !!!
后面的内容我们还是以Linux系统中的bash为例来介绍:)
变量的分类
Shell是一门动态类型语言和弱类型语言,我们可以把变量理解为KV对,key是变量名,value是变量值。变量大体可以分为环境变量
,系统变量
,用户定义的变量
三类。
- 环境变量
比如我们经常配置的JAVA_HOME
就属于环境变量,这些变量是所有Shell程序运行时都可以使用的变量。关于环境变量的操作命令举例如下:
使用export
命令定义的环境变量只在当前运行的shell进程中有效,结束进程就没了。所以我们要将配置变量定义在令小姐姐懵逼的一系列配置文件中,持久化下来。
说起配置文件,又不得不先提下shell程序和用户的Interactive和Login模式:)
- Interactive & Non-Interactive`
Interactive
通常是指读入写出数据都是从用户的terminal,也就是我们平时用命令行打开终端就是Interactive模式,而执行一个shell脚本就是Non-interactive
模式。怎么检验当前shell运行的模式是不是Interactive呢?小姐姐从GNU网站拷贝了一段装X脚本:
case "$-" in
*i*) echo This shell is interactive ;;
*) echo This shell is not interactive ;;
esac
结果如上所述。
- Login & Non-Login`
Login模式指的是用户成功登录后开启的shell进程,这时候会读取/etc/passwd
下用户所属的shell去执行。
Non-login模式指的是非登录用户状态下开启的shell进程,我们可以通过
echo $0
区分。
扯这么多是因为配置文件的加载顺序和shell进程是否运行在Interactive和Login模式有关系:D
这是阿姨从网上粘的图。bash支持的配置文件有/etc/profile,~/.bashrc等。
当调用一个Interactive&Login模式的shell进程时,配置文件的加载顺序为:
/etc/profile
—>( ~/.bash_profile, ~/.bash_login, ~/.profile)其中之一
—>~/.bash_loginout
(退出shell时调用)
当调用一个Interactive&non-Login模式的shell进程时,配置文件的加载顺序为:
/etc/bash.bashrc
—>~/.bashrc
当调用一个non-nteractive
模式的shell进程时,通常是执行脚本时,此时配置项是从环境变量中读取和执行的,也就是env
命令输出的配置项。
另外,在开启一个shell进程中,有一些参数的值也会影响到配置文件的加载。如--rcfile <file>,--norc等。这些参数的含义值可以使用man bash
进一步了解。只要保持默认值,其实就是我们上面介绍的配置文件加载顺序。
还有,在发行版的Linux系统中,Interactive&Login模式下的~/.bash_profile,
~/.bash_login, ~/.profile并不一定是三选一,看一下这三个脚本的内容会发现他们会继续调用下一个它想调用的配置文件,这样就可以避免配置项可能需要在不同的配置文件多次配置的弊端了。如centos7.2 中 ~/.bash_profile文件中实际调用了 ~/.bashrc文件。
按照模式和参数设置启动的shell程序的配置文件加载流程图如下:
好了,到目前我们总算把环境变量中配置文件的加载顺序理清了。下面列举一些常用的Shell环境变量吧。
-
系统变量
Shell中系统变量主要在对参数判断和命令返回值判断时使用,包括脚本和函数的参数和返回值判断。没啥可说的,主要难记且脚本中经常出现:
用户自定义的变量
是指我们在使用命令或脚本时定义的变量,因为shell是弱类型语言且语法XX,这里主要谈谈初学时的几个坑爹点:-
“=” 左右两边不能有空格
你懂我说的意思了吧。。
引用
所谓引用,指的是将字符串用引用符号包括起来,以防止其中的特殊符号被Shell解释为其他涵义。
常用的引用符号如下:
- $
前面我们其实一直在用 “ 的作用。
- shell中变量名的大小写是敏感的
好了就这么多吧。
条件测试的表达
shell脚本中除了变量,还经常出现的语法就是条件测试的判断。不会写脚本的开发小姐姐不是好运维,我们来一起侃侃吧。
- 基本语法
在Shell程序中,当指定的条件为真时,整个条件测试的返回值为 0;反之,如果指定的条件为假时,整个条件测试的返回值为 非0。
条件测试表达式的书写有test expression
和 [ expression ]
两种形式,注意后者的空格一定不能省!!
脚本中经常出现的有字符串测试、整数测试、文件测试、逻辑操作符测试。我们一起看下:)
-
字符串测试
注意:这里运算符 左右两边又一定要有空格了(下同),这样shell才能将之当成命令执行。
-
整数测试
-
文件测试
逻辑操作符测试
欢迎关注我的公众号:「码农知识点」,和我交流讨论~
参考资料:
[1].《Shell从入门到精通》
[2].https://www.edureka.co/blog/types-of-shells-in-linux/
[3].http://www.penguintutor.com/linux/basic-shell-reference
[4].https://apple.stackexchange.com/questions/361870/what-are-the-practical-differences-between-bash-and-zsh
[5].https://sunlightmedia.org/bash-vs-zsh/
[6].https://unix.stackexchange.com/questions/439042/debian-read-order-of-bash-session-configuration-files-inconsistent
[7].https://www.gnu.org/software/bash/manual/html_node/Bash-Startup-Files.html
[8].http://howtolamp.com/articles/difference-between-login-and-non-login-shell/
[9].https://shreevatsa.wordpress.com/2008/03/30/zshbash-startup-files-loading-order-bashrc-zshrc-etc/