摘要:Linux
,Shell
Shell条件测试要点总结
- Shell中条件测试能测试文件属性,字符串,整数大小等
- 条件测试语句为test,等同效果的测试符为
[]
,[[]]
,(())
- 条件判断语句为
if
,case
,case语句适合对预定义的固定变量进行匹配的场景 - test和[]效果等同,需要在
两端预留空格
- 在
test
和[]
引用变量或者调用命令执行结果的变量,尽量加双引号括起来,防止变量有空格报错参数过多 - 尽量使用
双中括号[[]]
代替单中括号[],双中括号中变量是一个整体不会被分割并且能兼容更多符号,单中括号需要加入更多的变量双引号和符号转义 - 双中括号[[]]对于字符串有
模式匹配
的判断功能 - 条件测试存在与或非逻辑&&,||,逻辑符号写在[]外,写在[[]]内
- 对于数值字符串判断可以使用
(())
的方式结合一般的数学判断符号进行使用
Shell条件测试语句
Shell的条件测试语句就是对条件进行判断,返回真和假,0表示真,非0表示假,和一般的编程语言相反,条件测试语句通常和if结合使用,实现不同判断结果执行不同操作的目的,主要包括:
-
文件测试
:判断文件是否存在,目录或文件判断,执行权限判断 -
整数测试
:判断整数之间的大小比较 -
字符串测试
:判断字符串长度是否为0,判断字符串是否相等 -
逻辑运算符
:实现与或非逻辑
条件测试语句的实现有test命令
,[]
,参数通用,速查如下
变量类型 | 操作符 | 说明 |
---|---|---|
文件 | -d | 是否是目录 |
文件 | -f | 是否是文件 |
文件 | -e | 是否存在该目录或者文件 |
文件 | -w | 文件是否可写 |
文件 | -r | 文件是否可读 |
文件 | -x | 文件是否是可执行 |
字符串 | = | 字符串是否相同 |
字符串 | != | 字符串是否不同 |
字符串 | > | 字符串大小比较,大于 |
字符串 | < | 字符串大小比较,小于 |
字符串 | -n | 字符串变量长度不为0,变量不为空 |
字符串 | -z | 字符串长度为0,变量为空 |
整数 | -gt | 是否大于 |
整数 | -ge | 是否大于等于 |
整数 | -lt | 是否小于 |
整数 | -le | 是否小于等于 |
整数 | -eq | 是否相等 |
整数 | -ne | 是否不等 |
组合逻辑 | -a | 与逻辑 |
组合逻辑 | -o | 或逻辑 |
组合逻辑 | ! | 非逻辑 |
在整数条件测试的时候可以使用(())
运算符,此时配合一般的数值比较符合(>,<,!=,==)即可
另一种常用的比较符是[[]]
,比[]更加通用,比[]增加字符串模式匹配,允许&&,||比较符
(1)true,false,$?
Shell和一般的编程语言对于true,false和1,0之间的转化是相反的,true是0,false是非0,在shell中,在linux系统中true和false在/bin目录下是可执行的命令,执行结果返回0和1,使用$?输出结果值
root@ubuntu:~# false
root@ubuntu:~# echo $?
1
root@ubuntu:~# true
root@ubuntu:~# echo $?
0
(2)文件测试
使用test或者[]进行文件条件测试,推荐使用[]
,不论是test还是[],符合和测试表达式之间都必须要有至少一个空格,两种方式的语句为
test <测试表达式>
[ <测试表达式> ]
文件测试的测试参数包括
操作符 | 说明 |
---|---|
-d | 是否是目录 |
-f | 是否是文件 |
-r | 是否可读 |
-w | 是否可写 |
-x | 是否可执行 |
-e | 文件或者目录是否存在 |
test语句只用案例,测试当前目录下的config.yml文件
test -f config.yml && echo 1 || echo 0 # 1
test -d confif.yml && echo 1 || echo 0 # 0
test -e config.yml && echo 1 || echo 0 # 1
test -x config.yml && echo 1 || echo 0 # 0
[]的使用案例
[ -f config.yml ] && echo 1 || echo 0 # 1
[ -r config.yml ] && echo 1 || echo 0 # 1
[ -e config.yml ] && echo 1 || echo 0 # 1
测试以下绝对路径和相对路径,都是可以实现判断功能
[ -e ../tmp ] && echo 1 || echo 0 #1
[ -e /home/tmp ] && echo 1 || echo 0 #1
[ -e ../abc.txt ] && echo 1 || echo 0 # 1
[ -d ../abc.txt ] && echo 1 || echo 0 # 0
[ -f ../abc.txt ] && echo 1 || echo 0 # 1
[ -f /home/abc.txt ] && echo 1 || echo 0 #1
test -f /opt/anaconda3/bin/python && echo 1 || echo 0 # 1,测试可执行文件
在Shell中为了能得到条件测试语句的返回值,使用&&
和||
进行输出,用法和Python的and,or类似,若前条件为真则输出and后面的结果,否则输出or后面的结果,从左到右进行接连判断
Shell >> a && b || c
Python >> a and b or c
(3)整数测试
整数测试主要测试数值间的大小关系,常用的用法是结合$?
判断上一个命令是否执行成功,整数测试同样可以使用test和[]比较符实现,语句为
test 整数 操作符 整数
[ 整数 操作符 整数 ]
整数比较的操作符参数如下
操作符 | 说明 |
---|---|
-gt | 大于 greater than |
-lt | 小于 lesser than |
-ge | 大于等于 greater equal |
-le | 小于等于 lesser equal |
-eq | 等于 equal |
-ne | 不等于 not equal |
[]运算符使用案例
a=3
[ $a -eq 3 ] && echo 1 || echo 0 # 1
let a=a+1
[ $a -eq 3 ] && echo 1 || echo 0 #0
[ $? -eq 0 ] && echo 1 || echo 0 # 0 上一个命令执行成功返回0
对于数值字符串的大小比较还可以使用C语言中的关系运算法配合(())
来使用,例子如下
root@ubuntu:~# ((1>0));echo $?
0
root@ubuntu:~# ((1>2));echo $?
1
root@ubuntu:~# ((1==2));echo $?
1
root@ubuntu:~# ((1!=2));echo $?
0
(4)字符串测试
字符串变量测试主要是比对字符串是否相同,字符串长度是否为0,使用test,[]运算符进行比较,测试参数如下
操作符 | 说明 |
---|---|
-z | 判断字符串长度是否为0 zero |
-n | 判断字符串变量长度是否非0 not zero |
!= | 判断字符串是否不等 |
= | 判断字符串是否不相同 |
使用示例如下
root@ubuntu:~# [ -z $JAVA_HOME ] && echo "请指定JAVA HONE" || echo JAVA_HOME是${JAVA_HOME}
JAVA_HOME是/opt/java/jdk1.8.0_60
root@ubuntu:~# test -z $JAVA_HOME;echo $?
1
可以使用[[]]实现更多字符串测试场景,比如判断字符串是否以某个字符串开头,在双中括号中配合通配符*
实现
root@ubuntu:~/Shell# [[ `python --version` == "Python 3.7"* ]];echo $?
0
root@ubuntu:~/Shell# [[ `python --version` = "Python 3.7"* ]];echo $?
0
root@ubuntu:~/Shell# [[ `python --version` = "Python 3.8"* ]];echo $?
1
在双中括号中,=和==都可以,但是单中括号不报错,但是判断错误
root@ubuntu:~/Shell# [ "`python --version`" = "Python 3.7"* ];echo $?
1 # 判断结果出错,应该是0
root@ubuntu:~/Shell# [[ "`python --version`" = "Python 3.7"* ]];echo $?
0 # 正确
备注:此处对``执行的结果加了双引号,否则报错,原因是命令直接结果包含空格,会迷惑bash shell识别为[: too many arguments
参数太多导致语法错误,不仅是使用命令输出结果,如果引用一个已有变量存在有空格的情况也会报错,解决方式使用双引号括起来作为一个变量,另一种方法是使用更通用的双中括号[[]]
,在双中括号中的字符串不会被空格分割,因此建议如下
- 在单中括号中,为了防止引用变量或者命令执行结果中有空格,使用双引号括起来
- 使用更通用的双中括号进行条件判断,双中括号中的变量不会根据空格分割
除了可以实现字符串的模式匹配(正则表达式)判断,配合=~
使用
root@ubuntu:~/Shell# [[ `python --version` =~ "Python 3."[0-9]"."[0-9] ]];echo $?
0
字符串除了判断是否相等还可以判断大小,字符串的大小按照字典顺序进行判断
root@ubuntu:~/Shell# [[ a > b ]];echo $?
1
root@ubuntu:~/Shell# [[ b > a ]];echo $?
0
也可以使用单中括号,此时需要在中括号内转义
root@ubuntu:~/Shell# [ a \> b ];echo $?
1
root@ubuntu:~/Shell# [ b \> a ];echo $?
0
(5)多条件组合测试
&&,||,!分别代表与或非,不能写在单个[]中,需要多个[]之间使用与或非连接,如果写在[]里面需要使用-a,-o,!
操作符 | 说明 |
---|---|
-a | 与 |
-o | 或 |
! | 非 |
使用[]比较符,使用||或者&&需要写在[]外面,使用-a,-o写在里面
root@ubuntu:~/Shell# [ -e test1.sh ] || [ -e test2.sh ];echo $?
1
root@ubuntu:~/Shell# [ -e test1.sh -o -e test2.sh ];echo $?
1
root@ubuntu:~/Shell# [ -e test1.sh ];echo $?
1
还可以使用双中括号[[]]
,&&和||操作符写在双中括号内部
root@ubuntu:~/Shell# [[ -e test1.sh || -e test2.sh ]];echo $?
1
root@ubuntu:~/Shell# [[ ! -e test2.sh ]];echo $?
0
if条件语句
if条件语句的语法如下
if [ 条件测试语句 ]
then
代码块
fi
或者将then和if语句卸载同一行,需要加;
隔开
if [ 条件测试语句 ];then
代码块
fi
简单编写一个if单分支语句
#!/bin/bash
if [ -n $JAVA_HOME ];then
echo $JAVA_HOME
fi
if的多分支语句使用else和elif,此处和Python类似,其中if和elif都需要接then,最后一个else不需要加then
#!/bin/bash
if [ -z $JAVA_HOME ];then
echo "请指定JAVA HOME"
else
echo $JAVA_HOME
fi
#!/bin/bash
python_version=`python --version`
if [[ $python_version = "Python 3.6"* ]];then
echo 'Python版本是3.6'
elif [[ $python_version = "Python 3.7"* ]];then
echo 'Python版本是3.7'
elif [[ $python_version = "Python3.8"* ]];then
echo 'Python版本是3.8'
else
echo 'Python版本未知'
fi
结合命令行传入位置参数再进行if判断,判断条件带有组合条件的案例
#!/bin/bash
val=$1
if [[ val -gt 3 ]];then
echo '输入值大于3'
elif [[ val -le 3 && val -gt 0 ]];then
echo '输入值小于等于3且大于0'
else
echo '输入值小于等于0'
fi
case条件语句
Shell中的case类似Java的switch case,适合判断条件表达式是预设定的多个或者一组枚举值的情况,句法如下
case 变量值 in
条件表达式1)
代码块1
;;
条件表达式2)
代码块2
;;
条件表达式3)
代码块3
;;
*)
无匹配代码块
esca
其中;;
双分号表示终止case选项,最后一个*)
表示非上面条件的其他值,在条件表达式中可以使用的匹配符主要有
匹配符 | 说明 |
---|---|
* | 任意字符 |
? | 任意单个字符 |
[] | 括号中的任意一个字符 |
| | 多重选择 |
在配合使用匹配符之后,相比于if语句能够达到代码结构精简的效果,比如
#!/bin/bash
read -p "我已确认遵守该协议并安装。" val
case "$val" in
y|Y|yes|ok|OK|YES|OK|1)
echo "开始执行安装"
;;
*)
echo "退出安装"
esac
如果上面这个不用case语句使用if则要写四种情况的or语句,十分复杂,case语句的另一种常用的方法是对预设置的固定变量进行判断
#!/bin/bash
case "$1" in
-h)
echo "hello"
;;
-b)
echo "beybey"
;;
-p)
echo "paste"
;;
-c)
echo "cate"
;;
*)
printf "没有设定参数'%s'\n" "$1"
esac
使用效果如下
root@ubuntu:~/Shell# bash test.sh -c
cate
root@ubuntu:~/Shell# bash test.sh -l
没有设定参数'-l'