Shell脚本的if语句、循环语句中都会有一个逻辑判断式。逻辑判断式用于各类条件的判断,除非程序是流水帐,不然是必定会用到逻辑判断的,可见其重要性。linux
Shell的逻辑判断,我目前所知共有两种:shell
一、test命令,能够结合命令执行结果变量($?)或者&&以及||来实现不一样条件走不一样分支bash
二、[ 条件判断 ],这种方式经常使用于if语句和while语句中less
shell中的逻辑判断
格式1:if 条件 ; then 语句; fi
格式2:if 条件; then 语句; else 语句; fi
格式3:if …; then … ;elif …; then …; else …; fi
逻辑判断表达式:if [ $a -gt $b ]; if [ $a -lt 5 ]; if [ $b -eq 10 ]等 -gt (>); -lt(<); -ge(>=); -le(<=);-eq(==); -ne(!=) 注意处处都是空格
可使用 && || 结合多个条件
if [ $a -gt 5 ] && [ $a -lt 10 ]; then
if [ $b -gt 5 ] || [ $b -lt 3 ]; then测试
#!/bin/bash # author:菜鸟教程 # url:www.runoob.com a=10 b=20 if [ $a -eq $b ] then echo "$a -eq $b : a 等于 b" else echo "$a -eq $b: a 不等于 b" fi if [ $a -ne $b ] then echo "$a -ne $b: a 不等于 b" else echo "$a -ne $b : a 等于 b" fi if [ $a -gt $b ] then echo "$a -gt $b: a 大于 b" else echo "$a -gt $b: a 不大于 b" fi if [ $a -lt $b ] then echo "$a -lt $b: a 小于 b" else echo "$a -lt $b: a 不小于 b" fi if [ $a -ge $b ] then echo "$a -ge $b: a 大于或等于 b" else echo "$a -ge $b: a 小于 b" fi if [ $a -le $b ] then echo "$a -le $b: a 小于或等于 b" else echo "$a -le $b: a 大于 b" fi
10 -eq 20: a 不等于 b
10 -ne 20: a 不等于 b
10 -gt 20: a 不大于 b
10 -lt 20: a 小于 b
10 -ge 20: a 小于 b
10 -le 20: a 小于或等于 bui
参数url |
功能spa |
说明.net |
-e命令行 |
文件是否存在 |
对文件类型的判断 test -e file |
-f |
判断文件名是否存在且为文件 |
|
-d |
判断文件名是否存在且为目录 |
|
-r |
判断对该文件是否有“可读”权限 |
对文件权限的检测 test -r file |
-w |
判断对该文件是否有“可写”权限 |
|
-x |
判断对该文件是否有“可执行”权限 |
|
-nt |
(newer than)判断file1是否比file2新 |
两个文件之间的比较 test file1 -nt file2 |
-ot |
(older than)判断file1是否比file2旧 |
|
-ef |
判断file1和file2是否为同一文件 |
|
-eq |
两数值是否相等(equal) |
两个整数之间的比较 test n1 -eq n2 |
-ne |
两数值是否不等(not equal) |
|
-gt |
n1大于n2(greater than) |
|
-lt |
n1小于n2(less than) |
|
-ge |
n1大于等于n2(greater than or equal) |
|
-le |
n1小于等于n2(less than or equal) |
|
test -z string |
判断字符串是否为空,若是为空,则为true |
字符串的判断 对参与判断的字符串,最好加上""",如"$var"这样的格式,否则会产生“参数过多”的错 判断相等的时候,"="和"=="是等效的 |
test -n string |
判断字符串是否为空,若是为空,则返回false |
|
test str1=str2 |
判断str1是否等于str2,若相等,则返回true |
|
test str1!=str2 |
判断str1是否不等于str2,若相等,则返回false |
|
-a |
两个条件同时成立则为真 test -r file1 -a test -x file2 |
多重条件的判断
|
-o |
两个条件任意一个成立则为真 test -r file1 -a test -x file2 |
|
! |
对测试条件结果取反 test ! -x file |
这是一些经常使用的test命令的参数,“[]”的用法与test命令相似,只要去掉test这个命令就行,其他不变。
运算符号 | 表明意义 |
= | 等于 应用于:整型或字符串比较 若是在[] 中,只能是字符串 |
!= | 不等于 应用于:整型或字符串比较 若是在[] 中,只能是字符串 |
< | 小于 应用于:整型比较 在[] 中,不能使用 表示字符串 |
> | 大于 应用于:整型比较 在[] 中,不能使用 表示字符串 |
-eq | 等于 应用于:整型比较 |
-ne | 不等于 应用于:整型比较 |
-lt | 小于 应用于:整型比较 |
-gt | 大于 应用于:整型比较 |
-le | 小于或等于 应用于:整型比较 |
-ge | 大于或等于 应用于:整型比较 |
-a | 双方都成立(and) 逻辑表达式 –a 逻辑表达式 |
-o | 单方成立(or) 逻辑表达式 –o 逻辑表达式 |
-z | 空字符串 |
-n | 非空字符串 |
关于&&和||
用&&和||结合多个条件
例如1:
[root@localhost shell]# cat if4.sh
#/bin/bash
a=5
if [ $a -gt 4 ] && [ $a -lt 6 ]
then
echo "4<a<6"
else
echo nook
fi
[root@localhost shell]# sh if4.sh
4<a<6
这个是用于联合两个命令的,逻辑与(&&)和逻辑或(||),在if和循环的判断式中的意义与C语言中是同样的。
逻辑与:表示两个同时为真,则改表达式为真,不然为假。
逻辑或:表示任意一个为真,则表达式为真,不然为假。
可是在test命令中,这两个操做符的意义有所 不一样
命令格式 |
解释 |
cmd1 && cmd2 | 若cmd1执行完毕且正确执行($?=0),则开始执行cmd2 若cmd1执行完毕且返回出错($?≠0),则不执行cmd2 |
cmd1 || cmd2 |
若cmd1执行完毕且正确执行($?=0),则不执行cmd2 若cmd1执行完毕且返回出错($?≠0),则执行cmd2 |
若是是多于两个命令的联合,执行结果会不断的日后传,影响后面的判断。同时由于$?只有一个,因此这个影响有个就近原则。
格式1:if条件;then语句;fi (若是...而后...)
例如:
[root@localhost shell]# cat if1.sh
#/bin/bash
a=3
if [ $a -gt 1 ]
then
echo ok
fi
结果:
[root@localhost shell]# sh if1.sh
ok
格式2:if条件;then语句;else语句;fi (若是...而后...;不知足条件...而后....)
示列:
[root@localhost shell]# cat if2.sh
#/bin/bash
a=1
if [ $a -gt 3 ]
then
echo ok
else
echo no ok
fi
结果:
[root@localhost shell]# sh if2.sh
nook
格式3:if...;then...;elif...;then...;else...;fi (若是...而后...;还有...而后...,不知足条件...而后...)
(能够添加多个elif)
例如:
[root@congji shell]# cat if3.sh
#/bin/bash
a=*
if [ $a -gt 6 ]
then
echo "a > 3"
elif
[ $a -lt 9 ]
then
echo "< 9"
else
echo no ok
fi
当a等于5时
[root@localhost shell]# sh if3.sh
< 9
当a等于7时
[root@localhost shell]# sh if3.sh
a > 3
因此当if知足条件,elif不会执行
test 命令
使用方法:test EXPRESSION
如:
[root@localhost ~]# test 1 = 1 && echo 'ok'
ok
[root@localhost ~]# test -d /etc/ && echo 'ok'
ok
[root@localhost ~]# test 1 -eq 1 && echo 'ok'
ok
[root@localhost ~]# if test 1 = 1 ; then echo 'ok'; fi
精简表达式
[] 表达式
[root@localhost ~]# [ 1 -eq 1 ] && echo 'ok'
ok
[root@localhost ~]# [ 2 < 1 ] && echo 'ok'
-bash: 2: No such file or directory
[root@localhost ~]# [ 2 \< 1 ] && echo 'ok'
[root@localhost ~]# [ 2 -gt 1 -a 3 -lt 4 ] && echo 'ok'
ok
[root@localhost ~]# [ 2 -gt 1 && 3 -lt 4 ] && echo 'ok'
-bash: [: missing `]'
注意:在[] 表达式中,常见的>,<须要加转义字符,表示字符串大小比较,以acill码 位置做为比较。 不直接支持<>运算符,还有逻辑运算符|| && 它须要用-a[and] –o[or]表示
[[]] 表达式
[root@localhost ~]# [ 1 -eq 1 ] && echo 'ok'
ok
[root@localhost ~]$ [[ 2 < 3 ]] && echo 'ok'
ok
[root@localhost ~]$ [[ 2 < 3 && 4 > 5 ]] && echo 'ok'
ok
注意:[[]] 运算符只是[]运算符的扩充。可以支持<,>符号运算不须要转义符,它仍是以字符串比较大小。里面支持逻辑运算符:|| &&
注意:全部字符 与逻辑运算符直接用“空格”分开,不能连到一块儿。
(1). if [ -z "$a" ] 这个表示当变量a的值为空时会怎么样 ,例如我如今须要获取一个文件内容的行数赋值给一个变量,而后把这个变量做为判断条件,可是我不肯定这个文件会否存在,因此我得先判断这个变量是否为空,为空则要打印错误。
if 判断文件、目录属性
[ -f file ]判断是不是普通文件,且存在
[ -d file ] 判断是不是目录,且存在
[ -e file ] 判断文件或目录是否存在
[ -r file ] 判断文件是否可读
[ -w file ] 判断文件是否可写
[ -x file ] 判断文件是否可执行
esle if和elif,是同样的
判断是否为目录时,else后是mkdir $d
#!/bin/bash # author:菜鸟教程 # url:www.runoob.com file="/var/www/runoob/test.sh" if [ -r $file ] then echo "文件可读" else echo "文件不可读" fi if [ -w $file ] then echo "文件可写" else echo "文件不可写" fi if [ -x $file ] then echo "文件可执行" else echo "文件不可执行" fi if [ -f $file ] then echo "文件为普通文件" else echo "文件为特殊文件" fi if [ -d $file ] then echo "文件是个目录" else echo "文件不是个目录" fi if [ -s $file ] then echo "文件不为空" else echo "文件为空" fi if [ -e $file ] then echo "文件存在" else echo "文件不存在" fi
文件可读
文件可写
文件可执行
文件为普通文件
文件不是个目录
文件不为空
文件存在
if判断的一些特殊用法
if [ -z “$a” ] 这个表示当变量a的值为空时会怎么样 ,判断变量是否存在。
if [ -n “$a” ] 表示当变量a的值不为空
if grep -q ‘123’ 1.txt; then 表示若是1.txt中含有’123’的行时会怎么样
if [ ! -e file ]; then 表示文件不存在时会怎么样
if (($a<1)); then …等同于 if [ $a -lt 1 ]; then…
[ ] 中不能使用<,>,==,!=,>=,<=这样的符号
=:是赋值;==:是等于号
不管是-n仍是-z,都不能做用在文件上。只能做用于变量。
#!/bin/bash # author:菜鸟教程 # url:www.runoob.com a="abc" b="efg" if [ $a = $b ] then echo "$a = $b : a 等于 b" else echo "$a = $b: a 不等于 b" fi if [ $a != $b ] then echo "$a != $b : a 不等于 b" else echo "$a != $b: a 等于 b" fi if [ -z $a ] then echo "-z $a : 字符串长度为 0" else echo "-z $a : 字符串长度不为 0" fi if [ -n $a ] then echo "-n $a : 字符串长度不为 0" else echo "-n $a : 字符串长度为 0" fi if [ $a ] then echo "$a : 字符串不为空" else echo "$a : 字符串为空" fi
abc = efg: a 不等于 b abc != efg : a 不等于 b -z abc : 字符串长度不为 0 -n abc : 字符串长度不为 0 abc : 字符串不为空
shell中的case判断
格式 case 变量名 in
value1)
command
;;
value2)
command
;;
*)
commond
;;
esac
在case程序中,能够在条件中使用|,表示或的意思, 好比 2|3) command ;;
实例:
查询指定的文件是否存在,不存在,就touch建立出现了
只须要把echo后面的$,换成你须要的,-d,-e,-r,-o等等
!vi命令:执行是一个vi编辑的文件。!sh命令:执行上一个sh执行的文件,也就是sh 3.sh
判断1.sh是否为空,为空输出ok.变量要用双引号引发来。文件就不用了。
判断$b的值是否为空,为空输出$b.不为空,输出b is null。
#!/bin/bash
f="/tmp/aminglinux"
[ -f $f ] || touch $f #若是上面的目录不存在,就touch建立他
if [ ! -f $f ] #前面添加一个叹号,取相反的意思,-f是存在的意思, ! -f是不存在
then
touch $f #touch能够
[ -f $f ] && rm -f $f
if [ -f $f]
rm -f $f
fi
if特殊用法
#!/bin/bash
n='wc -l /tmp/local'
if [ $n -gt 100 ] #统计文件大于100行
then
echo sdahd
fi
-n STRING
the length of STRING is nonzero
STRING equivalent to -n STRING
-z STRING
the length of STRING is zero
-n 判断字符串是否不为零
-z 判断字符串是否为零
tar 0是为了提示你输入的范围不在0-100之内;须要告诉用户。
咱们在介绍case这个命令以前,先解释一个read -p命令。
咱们先用命令行来测试一下这个命令。
[root@localhost ~]# read -p "Please input a number: " n
Please input a number: 123
[root@localhost~]# echo $n
123
经过上面的实验咱们能够发现,read -p 能够和用户实现一个交互,并赋给变量一个值。
删除笔记修改笔记
咱们等会写一个关于case的脚本。咱们把它分红2部分来解析。首先看第一部分:
#/bin/bash
read -p "Please input a number: " n
if [ -z "$n" ]
then
echo "Please input a number."
exit 1
fi
首先,咱们让用户输入一个数字,-z表示,若是变量为空,那么会返回一个Please input a number的字符串,而且exit退出,=。
还记得咱们编译时,若是失败了,当咱们用echo !$查询时,发现返回值为1,这里的1就是这个意思。
n1=`echo $n|sed 's/[0-9]//g'`
if [ -n "$n1" ]
then
echo "Please input a number."
exit 1
fi
若是用户输入的值不为空,那么跳入这一步,首先会把用户输入的变量拿出来,作一个过滤,sed命令把全部数字所有清空,-n表示若是sed清空数字后,字符不为空,则表示用户输入了一个不是数字的字符,则会返回,Please input a number!,这里加一个!和上面的作区分。
执行结果:
[root@localhost shell]# sh case1.sh
Please input a number:
Please input a number.
[root@localhost shell]# sh case1.sh
Please input a number: ds1
Please input a number!!
第二部分:使用了if...then...;elif...then...;eles....
若是变量是n大于等于0,小于等于60,则变量tag=1。
若是变量n大于等于60,小于等于80,则变量tag=2。
若是变量n大于等于80,小于等于90,则变量tag=3。
若是变量n大于等于90,小于等于100,则变量tag=4。
if [ $n -lt 60 ] && [ $n -ge 0 ]
then
tag=1
elif [ $n -ge 60 ] && [ $n -lt 80 ]
then
tag=2
elif [ $n -ge 80 ] && [ $n -lt 90 ]
then
tag=3
elif [ $n -ge 90 ] && [ $n -le 100 ]
then
tag=4
else
tag=0
fi
第三部分,就是咱们要说的case循环了.注意格式,case 变量 in ,1)...;; 2)...;; 3)...;;,最后结尾用esac,咱们也能够|来表示并列,例如,3|4),就是当变量等于3或者4时,返回下面的命令。
当变量tag=1时,返回not ok
当变量tag=2时,返回ok
当变量tag=3时,返回ook
当变量tag=4时,返回oook
当变量等于其余时,说明用户输入的值不在这个范围内,则返回值"The number range is 0-100."
case $tag in
1)
echo "not ok"
;;
2)
echo "ok"
;;
3)
echo "ook"
;;
4)
echo "oook"
;;
*)
echo "The number range is 0-100."
;;
esac
注意:
判断$a是否为文件,不是的话就touch,后来改为判断$a是否为目录,不是的话仍是touch一下,接着您就判断$a是否存在,不存在的话仍是touch一下。
正确逻辑应该是 mkdir
2.echo $?在脚本中须要放在exit以前?
echo $? 不须要放在脚本中的
常见问题:判断$a是否为文件,不是的话就touch,后来改为判断$a是否为目录,不是的话仍是touch一下,接着您就判断$a是否存在,不存在的话仍是touch一下。正确逻辑应该是 mkdir
二、出现参数太多 这个文件存在 -n 就不出现参数太多
答:先运行 wc -l /tmp/gao 看看结果
你比较的是一个数字,而不是 “数字 文件名”, n=`wc -l /tmp/gao|sed 's/[^0-9]//g'` 没有报错.
三、[ if -n ] 和 [ if -z ] 对文件判断是否为空并没用有
如图 "若是 999文件不为空, 则ok", 但我并无999文件
答:经证明,不管是-n仍是-z,都不能做用在文件上。只能做用于变量。
比较连接:https://blog.csdn.net/lovektm/article/details/79275375