十六周三次课

一、 shell脚本中的逻辑判断

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 



注意:[[]] 运算符只是[]运算符的扩充。可以支持<,>符号运算不须要转义符,它仍是以字符串比较大小。里面支持逻辑运算符:|| && 

 

注意:全部字符 与逻辑运算符直接用“空格”分开,不能连到一块儿。 


三、 if特殊用法(经常使用)

(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能够

fi

 

[ -f $f ] && rm -f $f

等于

if [ -f $f]

then

       rm -f $f

fi

 

 if特殊用法

#!/bin/bash

n='wc -l /tmp/local'

if [ $n -gt 100 ]               #统计文件大于100行

then 

       echo sdahd

fi 


四、case判断 

-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

相关文章
相关标签/搜索