十年运维系列之基础篇 - Linux
作者:曾林
联系:[email protected]
版权:文章未经同意请勿转载
一、使用if语句
通过shell,我们可以来写出这样的一个使用if语句的shell脚本片段。如下所示:
#!/bin/bash # test "if" statement x=5 if [ $x = 5 ]; then echo "x equals 5" else echo "x doesn‘t equals 5" fi
或者可以直接在命令行中输入以上代码(略有简化),如下图所示:
if语句的语法格式如下:
if commands; then commands [elif commands; then commands] [else commands] if
在这个语法格式中,“command”可以是一组命令。乍看上去可能会有些迷惑。在去除这个迷惑之前,我们必须要先了解一下shell如何判断一个命令的成功与失败。
二、退出状态
命令(包括我们编写的shell脚本和shell函数)在执行完毕后,会向操作系统发送一个值,称之为“退出状态”。这个值是一个0~255的整数,用来指示命令执行成功还是失败。按照惯例,数值0表示执行成功,其他的数值表示执行失败。shell提供了一个可以用来检测退出状态的参数。这个参数就是$?。如下图所示:
在这个例子中,我们两次执行了ls命令。第一次,命令执行成功,如果显示参数“$?”的值,可以看到值是0.第二次执行ls命令时,产生了一个错误,再次显示“$?”的值,这次值是2,表示这个命令遇到了一个错误。有些命令使用不同的退出值来诊断错误,而需要命令执行失败时,只是简单地退出并发送数字1.man手册中经常会包括一个标题为“Exit Status”的段落,它描述使用的代码。数字0总是表示执行成功。
shell提供了2个非常简单的内置命令,它们不做任何事情,除了以一个0或1退出状态来终止执行。“true”命令总是表示执行成功,而“false”命令总是表示执行失败。如下图所示:
我们可以用这两个命令来查看if语句是如何工作的。if语句真正做的事情是评估命令的成功或失败。
当在if后面的命令执行成功时,命令echo "it‘s true."会被执行,而当在if后面的命令执行失败时,该命令则不执行。如果在if后面有一系列的命令,那么则根据最后一个命令的执行结果来进行评估。如下图所示:
三、使用test命令
目前为止,经常和if一起使用的命令是test。test命令会执行各种检查和比较。这个命令有两种等价的形式:
test expression
以及更加流行的
[ expression ]
这里的expression是一个表达式,其结果是true或false。当这个表达式为true时,test命令会返回一个0退出状态;当表达式为false时,test命令的退出状态为1。
-
文件表达式(expression)
下表中的表达式展示的是有关文件的表达式。这些文件表达式用来评估文件的状态。
文件表达式 | 成为true的条件 |
file1 -ef file2 | file1和file2拥有相同的信息节点编号(这两个文件通过硬链接指向同一个文件) |
file1 -nt file2 | file1比file2新 |
file1 -ot file2 | file1比file2旧 |
-b file | file存在并且是一个块(设备)文件 |
-c file | file存在并且是一个字符(设备)文件 |
-d file | file存在并且是一个目录 |
-e file | file存在 |
-f file | file存在并且是一个普通文件 |
-g file | file存在并且设置了组ID |
-G file | file存在并且属于有效组ID |
-k file | file存在并且有“粘滞位(sticky bit)属性” |
-L file | file存在并且是一个符号链接 |
-O file | file存在并且属于有效用户ID |
-p file | file存在并且是一个命名管道 |
-r file | file存在并且可读(有效用户有可读权限) |
-s file | file存在并且其长度大于0 |
-S file | file存在并且是一个网络套接字 |
-t fd | fd是一个定向到终端/从终端定向的文件描述符,可以用来确定标准输入/输出/错误是否被重定向 |
-u file | file存在并且设置了setuid位 |
-w file | file存在并且可写(有效用户拥有可写权限) |
-x file | file存在并且可执行(有效用户拥有执行/搜索权限) |
下面代码展示了一个使用文件表达式的例子:
#!/bin/bash # test-file: Evaluate the status of a file FILE=~/.bashrc if [ -e "$FILE" ]; then if [ -f "$FILE" ]; then echo "$FILE is a regular file." fi if [ -d "$FILE" ]; then echo "$FILE is a directory." fi if [ -r "$FILE" ]; then echo "$FILE is readable." fi if [ -w "$FILE" ]; then echo "$FILE is writable." fi if [ -x "$FILE" ]; then echo "$FILE is executable/searchable." fi else echo "$FILE does not exist" exit 1 fi exit
关于上面的脚本,需要注意两个有趣的地方。首先,要注意$FILE在表达式内是如何被引用的。尽管引号不是必需的,但是这可以防范参数为空的情况。如果$FILE的参数扩展产生一个空值,将导致一个错误(操作符会被解释为非空的字符串,而不是操作符)。用引号把参数括起来可以确保操作符后面总是跟随一个字符串,即使字符串为空。其次,注意脚本末尾的exit命令。这个exit命令接受一个单独的可选参数,它将称为脚本的退出状态。当不传递参数时,退出状态默认为0。以这种方法使用exit命令,当$FILE扩展为一个不存在的文件名时,可以允许脚本提示失败。这个exit命令出现在脚本的最后一行。这样,当脚本执行到最后时,不管怎么样,默认情况下它将以退出状态0终止。
类似地,通过在return命令中包含一个整数参数,shell函数可以返回一个退出状态。如果要将上面的脚本转换为一个shell函数,从而能够在一个更大的程序中使用,可以将exit命令替换为return命令,并得到想要的行为。