要使Shell脚本程序具备一定的“智能”,面临的第一个问题就是如何区分不同的情况以确定执行何种操作。例如,当磁盘使用率超过95%时发送警告信息;当备份目录不存在时能够自动创建;当源码编译程序时若配置失败则不进行安装等。
Shell环境根据命令执行后的返回状态值来判断是否执行成功,当返回值为0时表示成功,否则表示失败或异常。使用专门的测试工具--test命令,可以对特定条件进行测试,并根据返回值来判断条件是否成立。
使用test测试命令时,包括以下两种形式。
格式1:test 条件表达式
格式2:[ 条件表达式 ]
这两种方式的作用完全相同,但通常后一种形式更为常用,也更贴近编程习惯。需要注意的是,方括号 [ ] 与条件表达式之间至少需要一个空格进行分隔。
根据需要测试的条件类别不同,条件表达式也不同。比较常见的条件操作包括文件测试、整数值比较、字符串比较,以及针对多个条件的逻辑测试,下面分别进行介绍。
文件传输指的是根据给定的路径名称,判断对应的是文件还是目录,或者判断文件是否可读、可写、可执行等。未经测试的常见操作选项如下,使用时将测试对象放在操作选项后即可。
-d:测试是否为目录(Directory)
-e:测试目录或文件是否存在(Exist)
-a:测试目录或文件是否存在(Exist)
-f:测试是否为文件(File)
-r:测试当前用户是否有权限读取(Read)
-w:测试当前用户是否有权限写入(Write)
-x:测试当前用户是否有权限执行(eXcute)
-L: 测试是否为软连接文件
执行条件测试操作后,通过预定义变量“$?”可以获得测试命令的返回状态值,从而判断该条件是否成立。例如,执行以下操作可测试目录/root/test是否存在,如果返回值为0,表示存在此目录,否则表示不存在或者虽然存在但不是目录。
[root@localhost ~]# [ -d /root/test ]
[root@localhost ~]# echo $?
1
[root@localhost ~]# ls /root
anaconda-ks.cfg guess.sh login.sh max min name.txt sum.sh test.sh xj.sh
通过查看变量“$?”变量值可以判断前一步的条件测试结果。但是操作比较烦琐,输出结果并不是很直观。为了更方便查看测试结果,可以结合命令分隔符“&&”和echo命令一起使用,当条件成立时直接输出yes。其中,“&&”符号表示“而且”的关系,只有当前面的命令支持成功后才会执行后面的命令。否则后面的命令将会被忽略。
将上面的改写一下:(无输出则表示不存在,输出yes表示该目录存在)
[root@localhost ~]# [ -d /root/test ] && echo "YES"
[root@localhost ~]# [ -d /etc/sysconfig ] && echo "YES"
YES
test命令中用于判断文件的选项有很多,从文件个数上分类的话,可以分为单个文件的判断和两个文件之间的比较。其中判断单个文件最常用的选项就是“-f”选项,在比较两个文件时,常用的选项有:
-nt:判断文件A是否比文件B新
-ot:判断文件A是否比文件B旧
-ef:判断两个文件是否是同一个文件,用来判断两个文件是否指向同一个inode号。
整数值比较是根据给定的两个整数值,判断第一个数与第二个数的关系,如是否大于,等于或者小于第2个数。
数值比较常用操作选项如下:
-eq:第一个数等于(Eqa) 第二个数。
-ne:第一个数不等于(NatEaqal第二个数。
-gt:第一个数大于(Greater Than) 第二个数。
-lt:第一个数小于Lesse Than) 第二个数。
-le: 第一个数小于或等于Lesse a Equal) 第二个数。
-ge: 第一个数大于或等于 (Greater a Equal) 第二个数。
整数值比较在Shell脚本编写中的应用较多。例如,用来判断已登录用户数量、开启进程数.磁盘使用率是否超标,以及软件版本号是否符合要求等,实际使用时,往往会通过变量引用.命令替换等方式来获取一个数值。
其通常用来检查用户输入,系统环境等是否满足条件,在提供交互式操作的Shell脚本中,也可用来判断用户输入的位置参数是否符合要求,字符串比较的常用操作选项如下。
=:第一个字符串与第二个字符串相同,
!=:第一个字符串与第二个字符串不相同, 其中!符号表示取反。
-z:检查字符串是否为空 (Zero).对于未定义或赋予空值的变量将视为空串。
例如,若要判断当前系统的语言环境,当发现不是’en.US" 时输出提示信息"Not en。US“,可以执行以下操作:
[root@localhost ~]# echo $LANG
zh_CN.UTF-8
[root@localhost ~]#[ $LANG != “en_US.UTF-8” ] && echo “not en.US”
not en.US
逻辑测试指的是判断两个或多个条件之间的依赖关系,当系统任务取决于多个不同的条件时,根据这些条件是否同时成立或者只要其中一个成立等情况,需要有一个测试的过程。常用的逻辑测试操作如下,使用时放在不同的测试语句或命令之间。
&&:逻辑与,表示"而且",只有当两个条件都成立时,整个测试命令的返回值才为0(结果成立)。使用Test命令测试时,“&&”可改为“-a”。
||:逻辑或,表示"或者",只要两个条件中有一个成立,整个测试命令的返回值就为0(结果成立)。使用Test命令测试时,“||”可改为“-o”
!:逻辑否,表示"不",只有当指定的条件不成立时,整个测试命令的返回值才为0(结果成立)。
在上述的逻辑测试的操作选项中,在实际应用中也用于间隔不同的命令操作,其作用也是相似的。如"make && make install"的编译安装操作。
在编写Shell脚本的时候,使用“&&”和“||”逻辑测试已经可以完成简单的判断并执行相应的操作,但是当需要选择执行的命令语句较多时,这种方式将使执行的代码复杂化,不好理解。而使用专门的if条件语句,可以更好的整理脚本结构,使得层次分明,清晰易懂。
在Shell脚本应用中,if语句是最为常用的一种流程控制方式,用来根据特定的条件测试结果,分别执行不同的操作。根据不同的复杂程度,if语句的选择结构可以分为三种基本类型,适用于不同的应用场合。
if语句的“分支”指的是不同测试结果所对应的执行语句(一条或多条)。对于单分支的选择结构,只有在“条件成立“时才会执行对应的代码,否则不执行任何操作。单分支的if语句的语法格式如下所示。
if 条件测试操作
then
命令序列
fi
在上述语句结构中,条件成熟操作即”[ 条件表达式 ]“语句,也可以是其它可执行的命令语句;命令序列指的是一条或多条可执行的命令行,也包括嵌套使用的if语句或其他流程控制语句。
单分支if语句的执行流程:首先判断条件测试的结果,如果返回值为0,表示条件成立,则执行then后面的命令序列,一直到遇见fi结束判断为止,继续执行其它脚本代码;如果返回值不为0,则忽略then后面的命令序列,直接跳转到fi行后面执行其它脚本代码。
对于双分支的选择结构,要求针对”条件成立“”条件不成立”两种情况分别执行不同的操作。双if语句的语法格式如下所示。
if 条件测试操作
then
命令序列 1
else
命令序列 2
fi
双分支if语句的执行流程:首先判断条件测试操作的结果,如果条件成立,则执行then后面的命令序列1,忽略else及后面的命令序列2,直到遇见fi结束判断;如果条件不成立,则忽略then及后面的命令序列1,直接跳到else后面的命令序列2并执行,直到遇见fi结束判断。
由于if语句可以根据测试结果的成立、不成立分别执行操作,所以能够嵌套使用,进行多次判断。例如,首先判断某学生的得分是否及格,若及格则再次判断是否高于90分等,多分支if语句的语法格式如下。
if 条件测试操作1
then
命令序列 1
elif 条件测试操作2
then
命令序列 2
else
命令序列 3
fi
上述语句结构中只嵌套了一个elif语句作为示例,实际上可以嵌套多个。if语句在编写shell脚本时并不常用,因为多重嵌套会使程序结构变得复杂。当确实需要多分支的程序结构时,采用case语句更加方便。
多分支 if 语句的执行流程:首先判断条件测试操作 1 的结果,如果条件 1 成立,则执行命令序列 1,然后跳至 fi 结束判断;如果条件 1 不成立,则继续判断条件测试操作 2 的结果,如果条件 2 成立,则执行命令序列 2,然后跳至 fi 结束判断……如果所有的条件都不满足,则执行 else 后面的命令序列 n,直到遇见 fi 结束判断。
#判断目录是否存在
#!/bin/bash
#date:2020-2-21
read -p "请输入需要判断的目录:" name
if [ ! -d $name ]
then
echo "$name目录不存在,正在创建..."
mkdir -p $name
echo "$name目录创建完成."
fi
echo "$name目录已存在,正在退出..."
#测试主机存活状态
#!/bin/bash
#date 2020-2-24
read -p "请输入一个ip:" ip
ping -c 1 $ip &>/dev/null
if [ $? -eq 0 ]
then
echo "ip主机存活"
else
echo "ip主机未存活,请尽快查看主机情况"
fi
#猜数字游戏
#!/bin/bash
NO=20
read -p "input your num: " num
if [ $NO -gt $num ]; then ### 判断输入的数字和原始数字的大小,gt表示大于
echo "你猜测的太小了"
elif [ $NO -lt $num ]; then ####判断输入的数字和原始数字的大小,lt表示小于
echo "你猜测的太大了"
else
echo "恭喜你猜对了"
fi