近来小姐姐又犯憨憨错误,问组内小伙伴 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/