条件测试操作
Shell 环境根据命令执行后的返回状态值($?)来判断是否执行成功,当返回值为 0 时表示成功,否则(非 0 值)表示失败或异常。使用专门的测试工具——test 命令,可以对特定条件进行测试,并根据返回值来判断条件是否成立(返回值为 0 表示条件成立)。
使用 test 测试命令时,包括以下两种形式。
test 条件表达式
或
[ 条件表达式 ]
文件测试
- -d:测试是否为目录(Directory)。
- -e:测试目录或文件是否存在(Exist)。
- -f:测试是否为文件(File)。
- -r:测试当前用户是否有权限读取(Read)。
- -w:测试当前用户是否有权限写入(Write)。
- -x:测试是否设置有可执行(Excute)权限。
执行条件测试操作以后,通过预定义变量$?可以获得测试命令的返回状态值,从而判断该条件是否成立。例如,执行以下操作可测试目录/media/是否存在,如果返回值$?为 0,表示存在此目录,否则表示不存在或者虽然存在但不是目录。
[root@localhost ~]# [ -d /media/ ]
[root@localhost ~]# echo $? //查看前一命令的返回值
0 //返回 0 表示条件成立
若测试的条件不成立,则测试操作的返回值将不为 0(通常为 1)。例如,执行以下操作展示了测试不存在目录的情况。
[root@localhost ~]# [ -d /media/cdrom/Server ]
[root@localhost ~]# echo $? //查看前一命令的返回值
1 //返回 1 表示条件不成立
为了更直观地查看测试结果,可以结合命令分隔符“&&”和 echo 命令一起使用,当条件成立时直接输出“YES”。其中,“&&”符号表示“而且”的关系,只有当前面的命令执行成功后才会执行后面的命令,否则后面的命令将会被忽略。例如,上述目录测试操作可以改写如下。
[root@localhost ~]# [ -d /media/cdrom/Server ] && echo "YES" //无输出表示该目录不存在
[root@localhost ~]# [ -d /media/cdrom ] && echo "YES" //输出"YES"表示该目录存在
YES
整数值比较
- -eq:第一个数等于(Equal)第二个数。
- -ne:第一个数不等于(Not Equal)第二个数。
- -gt:第一个数大于(Greater Than)第二个数。
- -lt:第一个数小于(Lesser Than)第二个数。
- -le:第一个数小于或等于(Lesser or Equal)第二个数。
- -ge:第一个数大于或等于(Greater or Equal)第二个数。
整数值比较在 Shell 脚本编写中的应用较多。例如,用来判断已登录用户数量、开启进程数、磁盘使用率是否超标,以及软件版本号是否符合要求等。实际使用时,往往会通过变量引用、命令替换等方式来获取一个数值。
例如,若要判断物理内存(Mem)当前的磁盘缓存(buff/cache)大小,当低于 1024MB时输出具体数值,可以执行以下操作。其中,“free -m”命令表示以 MB 为单位输出内存信息,提取的空闲内存数值通过命令替换赋值给变量 FreeCC。
[root@localhost ~]# FreeCC=$(free -m | grep "Mem: " | awk '{print $6}')
[root@localhost ~]# [ $FreeCC -lt 1024 ] && echo ${FreeCC}MB
275MB
字符串比较
字符串比较的常用操作
- =:第一个字符串与第二个字符串相同。
- !=:第一个字符串与第二个字符串不相同,其中“!”符号表示取反。
- -z:检查字符串是否为空(Zero),对于未定义或赋予空值的变量将视为空串。
例如,若要判断当前系统的语言环境,当发现不是“en.US”时输出提示信息“Not en.US”,
可以执行以下操作。
[root@localhost ~]# echo $LANG \\当前的语言环境
zh_CN.UTF-8
[root@localhost ~]# [ LANG != "en.US" ] && echo "not en.US" \\字符串比较测试 !=前后有空格
not en.US
[root@localhost ~]# read -p "是否覆盖现有文件(yes/no)?" ACK
是否覆盖现有文件(yes/no)?yes
[root@localhost ~]# [ $ACK = "yes" ] && echo "覆盖"
覆盖
[root@localhost ~]# read -p "是否覆盖现有文件(yes/no)?" ACK
是否覆盖现有文件(yes/no)?no
[root@localhost ~]# [ $ACK = "no" ] && echo "不覆盖"
不覆盖
逻辑测试
逻辑测试指的是判断两个或多个条件之间的依赖关系。常用的逻辑测试操作如下,使用时放在不同的测试语句或命令之间。
- &&:逻辑与,表示“而且”,只有当前后两个条件都成立时,整个测试命令的返回值
- 才为 0(结果成立)。使用 test 命令测试时,“&&”可改为“-a”。
- ||:逻辑或,表示“或者”,只要前后两个条件中有一个成立,整个测试命令的返回值即为 0(结果成立)。使用 test 命令测试时,“||”可改为“-o”。
- !:逻辑否,表示“不”,只有当指定的条件不成立时,整个测试命令的返回值才为 0(结果成立)。
例如,若要判断当前 Linux 系统的内核版本是否大于 3.4,可以执行以下操作。其中,内核版本号通过 uname 和 awk 命令获得。
[root@localhost ~]# uname -r //查看内核版本信息
3.10.0-514.el7.x86_64
[root@localhost ~]# Mnum=$(uname -r | awk -F. '{print $1}') //取主版本号
[root@localhost ~]# Snum=$(uname -r | awk -F. '{print $2}') //取次版本号
[root@localhost ~]# [ $Mnum -ge 3 ] && [ $Snum -gt 4 ] && echo "符合要求"
符合要求
if 条件语句
1. 单分支 if 语句
if 语句的“分支”指的是不同测试结果所对应的执行语句(一条或多条)。对于单分支的选择结构,只有在“条件成立”时才会执行相应的代码,否则不执行任何操作。单分支 if 语句的语法格式如下所示。
if 条件测试操作
then
命令序列
fi
2. 双分支 if 语句
对于双分支的选择结构,要求针对“条件成立”“条件不成立”两种情况分别执行不同的操作。双分支 if 语句的语法格式如下所示。
if 条件测试操作
then
命令序列 1
else
命令序列 2
fi
双分支 if 语句的执行流程:首先判断条件测试操作的结果,如果条件成立,则执行 then后面的命令序列 1,忽略 else 及后面的命令序列 2,直到遇见 fi 结束判断;如果条件不成立,则忽略 then 及后面的命令序列 1,直接跳至 else 后面的命令序列 2 并执行,直到遇见 fi 结束判断。
3. 多分支 if 语句
由于 if 语句可以根据测试结果的成立、不成立分别执行操作,所以能够嵌套使用,进行多次判断。例如,首先判断某学生的得分是否及格,若及格则再次判断是否高于 90 分等。多分支 if 语句的语法格式如下。
if 条件测试操作 1
then
命令序列 1
elif 条件测试操作 2
then
命令序列 2
else
命令序列 3
fi
多分支 if 语句的执行流程:首先判断条件测试操作 1 的结果,如果条件 1 成立,则执行命令序列 1,然后跳至 fi 结束判断;如果条件 1 不成立,则继续判断条件测试操作 2 的结果,如果条件 2 成立,则执行命令序列 2,然后跳至 fi 结束判断……如果所有的条件都不满足,则执行 else 后面的命令序列 n,直到遇见 fi 结束判断。
case 语句的结构
case 语句主要适用于以下情况:某个变量存在多种取值,需要对其中的每一种取值分别执行不同的命令序列。这种情况与多分支的 if 语句非常相似,只不过 if 语句需要判断多个不同的条件,而 case 语句只是判断一个变量的不同取值。
case 分支语句的语法结构如下所示。
case 变量值 in
模式 1)
命令序列 1
;;
模式 2)
命令序列 2
;;
……
* )
默认命令序列
esac
case 语句的执行流程:首先使用“变量值”与模式 1 进行比较,若取值相同则执行模式 1后的命令序列,直到遇见双分号“;;”后跳转至 esac,表示结束分支;若与模式 1 不相匹配,则继续与模式 2 进行比较,若取值相同则执行模式 2 后的命令序列,直到遇见双分号“;;”后跳转至 esac,表示结束分支……依此类推,若找不到任何匹配的值,则执行默认模式“*)”后的命令序列,直到遇见 esac 后结束分支。
case 分支语句的结构
- 使用 case 分支语句时,有几个值得注意的特点如下所述。
- case 行尾必须为单词“in”,每一模式必须以右括号“)”结束。
- 双分号“;;”表示命令序列的结束。
- 模式字符串中,可以用方括号表示一个连续的范围,如“[0-9]”;还可以用竖杠符号“|”表示或,如“A|B”。
- 最后的“*)”表示默认模式,其中的*相当于通配符。