shell编程之测试和判断
一、测试
程序运行中经常需要根据实际情况来运行特定的命令或代码段。比如判断某个文件或目录是否存在,如果文件或目录不存在,可能首先创建文件或目录。举例说,要判断文件/var/log/mlocate文件是否存在,可以先ls该文件,然后用$?来判断,如下所示:
#一个存在的文件
[root@Cfhost-170820-UCNK ~]# ls /var/lib/mlocate/
#如果ls成功,则$?返回值为0,说明该文件存在
[root@Cfhost-170820-UCNK ~]# echo $?
0
#ls一个不存在的文件,命令本身就会报错
[root@Cfhost-170820-UCNK ~]# ls /var/lib/mlocate2/
ls: cannot access /var/lib/mlocate2/: No such file or directory
#这里的$?的返回值是非0的,在不考虑文件权限的情况下,返回非0说明该文件不存在
[root@Cfhost-170820-UCNK ~]# echo $?
2
1.测试结构
测试的第一种使用方式是直接使用test命令,该命令格式如下:
test expression
其中expression是一个表达式,可以是算数比较、字符比较、文本和文件属性比较等。
第二种测试方式是用"["启动一个测试,再写expression,再以"]"结束测试。需要注意的是,左边的括号"["后有个空格,右括号“]"前面有一个空格,如果任意一边少空格都会造成shell报错。为增加代码的可读性,推荐使用第二种方式,而且这种方式更容易与if、case、while这些条件判断的关键张连用,该测试结构如下:
[ expression ]
2.文件测试
文件测试方法一
test file_operator FILE
文件测试方法二
[ file_operator FILE ]
例如:
#测试一个存在的文件,返回0
[root@Cfhost-170820-UCNK ~]# test -e /var/log/messages
[root@Cfhost-170820-UCNK ~]# echo $?
0
#测试一个不存在文件$?返回值不为0[
root@Cfhost-170820-UCNK ~]# test -e /var/log/messages01
[root@Cfhost-170820-UCNK ~]# echo $?
1
#用[]测试
[root@Cfhost-170820-UCNK ~]# [ -e /var/log/messages ]
[root@Cfhost-170820-UCNK ~]# echo $?
0
[root@Cfhost-170820-UCNK ~]# [ -e /var/log/messages01 ]
[root@Cfhost-170820-UCNK ~]# echo $?
1
测试某个文件的读、写、执行属性的代码:
[root@Cfhost-170820-UCNK ~]# cat rwx.sh
#!/bin/bash
read -p "What file do you want to test?" filename
if [ ! -e "$filename" ];then
echo "The file does not exits."
exit 1
fi
if [ ! -r "$filename" ];then
echo "$filename is readable."
fi
if [ ! -w "$filename" ];then
echo "$filename is writeable."
fi
if [ ! -x "$filename" ];then
echo "$filename is executable."
fi
3.字符串测试
#定义空字符串str1
[root@Cfhost-170820-UCNK ~]# str1=""
#测试str1是否为空,为空则返回0
[root@Cfhost-170820-UCNK ~]# test -z "$str1"
[root@Cfhost-170820-UCNK ~]# echo $?
0
#测试str1是否非空,非空则返回0,为空返回非0,此处返回1
[root@Cfhost-170820-UCNK ~]# test -n "$str1"
[root@Cfhost-170820-UCNK ~]# echo $?
1
#定义非空字符串str2,值为hello
[root@Cfhost-170820-UCNK ~]# str2="hello"
#测试str2是否为空,为空返回0,不为空返回非0,此处返回1
[root@Cfhost-170820-UCNK ~]# [ -z "$str2" ]
[root@Cfhost-170820-UCNK ~]# echo $?
1
#测试str2是否非空,非空返回0
[root@Cfhost-170820-UCNK ~]# [ -n "$str2" ]
[root@Cfhost-170820-UCNK ~]# echo $?
0
#比较str1和str2是否相同,相同返回0,否则返回非0,此处返回1
[root@Cfhost-170820-UCNK ~]# [ "$str1" = "$str2" ]
[root@Cfhost-170820-UCNK ~]# echo $?
1
#比较str1和str2是否不同,不同返回0,否则返回非0
[root@Cfhost-170820-UCNK ~]# [ "$str1" != "$str2" ]
[root@Cfhost-170820-UCNK ~]# echo $?
0
#比较str1和str2的大小,需要注意的是,>和<都需要进行转义
[root@Cfhost-170820-UCNK ~]# [ "$str1" \> "$str2" ]
[root@Cfhost-170820-UCNK ~]# echo $?
1
[root@Cfhost-170820-UCNK ~]# [ "$str1" \< "$str2" ]
[root@Cfhost-170820-UCNK ~]# echo $?
0
#如果不想用转义符,则可以用[ [ ] ]括起表达式
[root@Cfhost-170820-UCNK ~]# [[ "$str1" > "$str2" ]]
[root@Cfhost-170820-UCNK ~]# echo $?
1
[root@Cfhost-170820-UCNK ~]# [[ "$str1" < "$str2" ]]
[root@Cfhost-170820-UCNK ~]# echo $?
0
4.整数比较
返回0表示真,否则为假
[root@Cfhost-170820-UCNK ~]# num1=10
[root@Cfhost-170820-UCNK ~]# num2=10
[root@Cfhost-170820-UCNK ~]# num3=9
[root@Cfhost-170820-UCNK ~]# num4=11
[root@Cfhost-170820-UCNK ~]# [ "$num1" -eq "$num2" ]
[root@Cfhost-170820-UCNK ~]# echo $?
0
[root@Cfhost-170820-UCNK ~]# [ "$num1" -gt "$num2" ]
[root@Cfhost-170820-UCNK ~]# cho $?
-bash: cho: command not found
[root@Cfhost-170820-UCNK ~]# [ "$num1" -gt "$num2" ]
[root@Cfhost-170820-UCNK ~]# echo $?
1
[root@Cfhost-170820-UCNK ~]# [ "$num1" -eq "$num2" ]
[root@Cfhost-170820-UCNK ~]# echo $?
0
[root@Cfhost-170820-UCNK ~]# [ "$num1" -gt "$num3" ]
[root@Cfhost-170820-UCNK ~]# echo $?
0
[root@Cfhost-170820-UCNK ~]# [ "$num1" -lt "$num4" ]
[root@Cfhost-170820-UCNK ~]# echo $?
0
[root@Cfhost-170820-UCNK ~]# [ "$num1" -ge "$num2" ]
[root@Cfhost-170820-UCNK ~]# echo $?
0
[root@Cfhost-170820-UCNK ~]# [ "$num1" -le "$num2" ]
[root@Cfhost-170820-UCNK ~]# echo $?
0
[root@Cfhost-170820-UCNK ~]# [ "$num1" -ne "$num3" ]
[root@Cfhost-170820-UCNK ~]# echo $?
0
5.逻辑测试符和逻辑运算符
逻辑测试用于连接多个测试条件,并返回整个表达式的值。逻辑测试主要有逻辑非、逻辑与、逻辑或3种。
逻辑测试符:
!expression 如果expression为真,则测试结果为假
expression1 -a expression2 expression1和expression2同时为真,则测试结果为真
expression2 -o expression2 expression1和expression2只要有一个为真,则测试结果为真
举例说明:
#测试值为真的表达式在使用逻辑非后,表达式变为假,反之亦然
[root@Cfhost-170820-UCNK ~]# [ ! -e /var/lib/messages/ ]
[root@Cfhost-170820-UCNK ~]# echo $?
1
#表达式都为真,整个表达式才返回真,否则返回假
[root@Cfhost-170820-UCNK ~]# [ -e /var/lib/messages -a -e /var/lib/messages01 ]
[root@Cfhost-170820-UCNK ~]# echo $?
1
#测试表达式中只要有真,则整个表达式返回真
[root@Cfhost-170820-UCNK ~]# [ -e /var/lib/messages -o -e /var/lib/messages01 ]
[root@Cfhost-170820-UCNK ~]# echo $?
0
逻辑运算符
! 逻辑非,对真假取反
&& 逻辑与,连接两个表达式,只有两个表达式为真,结果才为真
|| 逻辑或,连接两个表达式,只要有一个表达式为真结果就为真
[root@Cfhost-170820-UCNK ~]# ! [ -e /var/lib/messages/ ]
[root@Cfhost-170820-UCNK ~]# echo $?
1
[root@Cfhost-170820-UCNK ~]# [ -e /var/lib/messages ] && [ -e /var/lib/messages01 ]
[root@Cfhost-170820-UCNK ~]# echo $?
1
[root@Cfhost-170820-UCNK ~]# [ -e /var/lib/messages ] || [ -e /var/lib/messages01 ]
[root@Cfhost-170820-UCNK ~]# echo $?
0
不管是逻辑运算符还是逻辑测试符,在做逻辑与、逻辑或、逻辑非运算时·都有共同的特点。比如逻辑与,由于需要所有的表达式都为0才返回0,因此在计算expression1结果为0后才会进行expression2的计算,如果expression2返回0则会进行expression3的计算,如果在计算过程中任何一部分的计算值都为0,则不会在计算后面的表达式。如果一开始就计算出expression1为非0,则可以断言整个表达式一定是返回非0,就没有必要计算expression2和expression3了。
#逻辑与运算
expression1 && expression2 && expression3
#逻辑与测试
[ expression1 -a expression2 -a expression3 ]
而对逻辑或来说,只要有一个表达式返回0,则可以断言整个表达式返回值为0,所以如果计算expression1的值为0,就不用再进行expression2和expression3的计算了。
#逻辑或运算
expression1 || expression2 || expression3
#逻辑或测试
[ expression1 -o expression2 -o expression3 ]
在实践过程中,常会将逻辑与、逻辑或的这些特点联合起来使用,使用方式如下:
expression && DoWhenExpressionTrue || DoWhenExpressionFalse
在这段代码中,从左到右分别用&&、||连接,这时,Shell首先计算expression,并返回其值。如果返回真,则会继续执行&&后的代码DoWhenExpressionTrue,如果该语句执行成功,则expression && DoWhenExpressionTrue整体返回假,则跳过DoWhenExpressionTrue,这时由于expression && DoWhenExpressionTrue 整体返回假,则用||连接的代码段DoWhenExpressionFalse一定会执行。
二、判断
有了测试,就要有获得测试结果的机制,并根据测试结果运行不同的代码段,这样程序就可以从简单的命令罗列变得更智能一些,从而实现程序的流程控制。在Shell中,流程控制分为两大类,一类是“循环",一类是”判断选择"。属于“循环”的有for、while、until。
1.if判断结构
[root@Cfhost-170820-UCNK ~]# cat score01.sh
#!/bin/bash
echo -n "Please input a score:"
read SCORE
if [ "$SCORE" -lt 60 ];then
echo "C"
fi
if [ "$SCORE" -lt 80 -a "$SCORE" -ge 60 ];then
echo "B"
fi
if [ "$SCORE" -ge 80 ];then
echo "A"
fi
执行结果如下:
[root@Cfhost-170820-UCNK ~]# sh score01.sh
Please input a score:95
A
[root@Cfhost-170820-UCNK ~]# sh score01.sh
Please input a score:75
B
[root@Cfhost-170820-UCNK ~]# sh score01.sh
Please input a score:45
C
2.if/else判断结构
[root@Cfhost-170820-UCNK ~]# cat check_file.sh
#!/bin/bash
FILE=/var/log/messages
if [ -e $FILE ];then
echo "$FILE exists"
else
echo "$FILE not exists"
fi
#当该文件存在时,执行结果如下
[root@Cfhost-170820-UCNK ~]# sh check_file.sh
/var/log/messages exists
3.if/elif/else判断结构
#在原来的score_01的文件基础上进行改进,这种判断结构更好的适用于多条件判断
[root@Cfhost-170820-UCNK ~]# cat score03.sh
#!/bin/bash
echo -n "Please input a score:"
read SCORE
if [ "$SCORE" -lt 60 ];then
echo "C"
elif [ "$SCORE" -lt 80 -a "$SCORE" -ge 60 ];then
echo "B"
else
echo "A"
fi
4.case判断结构
和if/elif/else判断一样,case判断结构也可以用于多种可能情况下的分支判断
#简单判断用户输入是小写还是大写或是数字
[root@Cfhost-170820-UCNK ~]# cat datect_input.sh
#/bin/bash
read -p "Give me a word:" input
echo -en "You gave me some"
case $input in
*[[:lover:]]*) echo -en "Lowercase " ;;
*[[:upper:]]*) echo -en "Uppercase " ;;
*[[:digit:]]*) echo -en "Numerical " ;;
*) echo "unknown input." ;;
esac