shell编程之条件语句(条件测试、if语句、case语句)

1,692 阅读8分钟

要使Shell脚本程序具备一定的“智能”,面临的第一个问题就是如何区分不同的情况以确定执行何种操作。Shell环境根据命令执行后的返回状态值($?)来判断是否执行成功,当返回值为0时表示成功,否则(非0值)表示失败或异常。

使用专门的测试工具——test命令,可以对特定条件进行测试,并根据返回值来判断条件是否成立(返回值为0表示条件成立)。

使用test测试命令时,有以下两种形式:

 格式一: test 条件表达式
 ​
 格式二: [ 条件表达式 ]
 ​
 #注意:每一项之间要有空格,格式二更常用
复制代码

条件测试:判断某需求是否满足,需要由测试机制来实现,专用的测试表达式需要由测试命令辅助完成。

测试过程,实现评估布尔声明,以便用在条件性环境下进行执行:

  • 若为真,则状态码变量 $? 返回0
  • 若为假,则状态码变量 $? 返回非0

1 条件测试语句

1.1文件测试

文件测试格式

 格式一: test 操作符 表达式
 ​
 格式二: [ 操作符 表达式 ]  
复制代码

常用的文件测试操作符:

操作符作用
-d测试文件是否为目录文件(Directory)
-f测试文件是否为普通文件(File)
-e测试目录或文件是否存在(Exist)。建议使用-e,-a可能不准确。
-a测试目录或文件是否存在(Exist)
-r测试当前用户是否有权限读取(Read)
-w测试当前用户是否有权限写入(Write)
-x测试当前用户是否有权限执行(eXcute)
-L测试是否为符号链接(软链接文件)
-nt判断文件A是否比文件B新
-ot判断文件A是否比文件B旧
-ef判断两个文件是否为同一文件。可用于硬链接,主要判断两个文件是否指向同一个inode。

属性测试补充:

操作符作用
-s FILE判断文件是否存在且非空
-t fdfd 指定文件描述符是否在某终端已经打开
-N FILE文件自从上一次被读取之后是否被修改过
-O FILE当前有效用户是否为文件属主
-G FILE当前有效用户是否为文件属组

示例1:

 [root@yuji ~]# test -d /etc/      //测试是否为目录
 [root@yuji ~]# echo $?
 0                                 //状态码返回0,表示为真,是目录
 [root@yuji ~]# [ -d /etc/ ]       //测试是否为目录
 [root@yuji ~]# echo $?
 0
 [root@yuji ~]# test -f /etc/sysconfig     //测试是否为普通文件
 [root@yuji ~]# echo $?
 1                                        //状态码返回1,表示为假,不是普通文件
 [root@yuji ~]# [ -f /etc/sysconfig ]     //测试是否为普通文件
 [root@yuji ~]# echo $?
 1
复制代码

1文件测试1.png

示例2:

-e和-a都可以测试文件是否存在,但和 !(取反)连用时,-a可能不准确,建议使用-e

 [root@yuji ~]# [ -e /etc/shadow ]     //测试文件是否存在
 [root@yuji ~]# echo $?
 0
 [root@yuji ~]# [ -a /etc/shadow ]     //测试文件是否存在
 [root@yuji ~]# echo $?
 0
 [root@yuji ~]# [ ! -a /etc/shadow ];echo $?     //测试文件是否不存在,判断结果有误
 0
 [root@yuji ~]# [ ! -e /etc/shadow ];echo $?     //测试文件是否不存在
 1
复制代码

1文件测试2.png

示例3:

测试root用户对文件是否拥有读、写、执行权限。

发现root用户的读取和写入权限,不受文件的基本权限控制;但执行权限受文件基本权限控制。

 [root@yuji ~]# ll /etc/shadow
 ----------. 1 root root 3792 3月  31 21:59 /etc/shadow
 [root@yuji ~]# [ -r /etc/shadow ]     //测试root用户对文件是否拥有读取权限
 [root@yuji ~]# echo $?
 0
 [root@yuji ~]# [ -w /etc/shadow ]     //测试root用户对文件是否拥有写入权限
 [root@yuji ~]# echo $?
 0
 [root@yuji ~]# [ -x /etc/shadow ]     //测试root用户对文件是否拥有执行权限
 [root@yuji ~]# echo $?
 1
复制代码

1文件测试3.png

示例4:

测试软链接时,目录名称后面不要加"/",不然会被判断为目录文件。

 [root@yuji /]# ls -dl /bin
 lrwxrwxrwx. 1 root root 7 3月   8 19:13 /bin -> usr/bin    //该文件为软链接文件
 [root@yuji /]# [ -L /bin ]        //测试是否为软链接,是
 [root@yuji /]# echo $?
 0
 [root@yuji /]# [ -L /bin/ ]       //测试是否为软链接,不是
 [root@yuji /]# echo $?
 1
 [root@yuji /]# ls -dl /bin/
 dr-xr-xr-x. 2 root root 49152 4月   5 02:15 /bin/      //该文件为目录文件
复制代码

1文件测试4.png

1.2 整数测试

整数值比较:

 [  整数变量1  操作符  整数变量2  ] 
 ​
 test 整数变量1  操作符  整数变量2
复制代码

操作符:

操作符含义
-eq等于(Equal)
-ne不等于(Not Equal)
-gt大于(Greater Than)
-lt小于(Lesser Than)
-le小于或等于(Lesser or Equal)
-ge大于或等于(Greater or Equal)

示例1:

 [root@yuji /]# [ 2 -lt 3 ]       //测试2是否小于3
 [root@yuji /]# echo $?
 0
 [root@yuji /]# a=5
 [root@yuji /]# b=6
 [root@yuji /]# [ $a -eq $b ]     //测试变量a是否等于变量b
 [root@yuji /]# echo $?
 1
复制代码

2整数测试1.png

示例2:

查询当前目录下的文件数量是否大于10,如果大于,则进行提示。

 [root@yuji ~]# ls | wc -l
 40
 [root@yuji ~]# test `ls |wc -l` -gt 10 && echo "文件数量大于10"
 文件数量大于10
复制代码

2整数测试2.png

1.3 字符串测试

常用的测试操作符

  • =:字符串内容相同
  • !=:字符串内容不同,! 号表示相反的意思
  • -z:字符串内容为空
  • -n:字符是否存在

字符串比较,常用的四种格式:

 [ "字符串1"  =  "字符串2"  ]   //测试字符串1和字符串2的内容是否相同
 [ "字符串1" !=  "字符串2”  ]   //测试字符串1和字符串2的内容是否不同
 [ -z "字符串" ]    //测试字符串是否为空,或者指定的变量是否为空值。(测试字符串的长度是否为零)
 [ -n "字符串" ]    //测试字符串是否存在,即是否不为空。(测试字符串的长度是否非零)
复制代码

示例1:

 [root@yuji ~]# str1=aa
 [root@yuji ~]# str2=bb
 [root@yuji ~]# [ $str1 = $str2 ];echo $?    //测试变量1和变量2的值是否相同
 1
 [root@yuji ~]# [ $USER = root ];echo $?     //测试当前用户是否为root
 0
 [root@yuji ~]# [ $USER != root ];echo $?    //测试当前用户是否不为root
 1

 [root@yuji ~]# [ -z $HOME ];echo $?        //测试变量HOME是否为空值
 1
复制代码

3字符串1.png

示例2:

使用-z或-n测试变量时,变量需要加引号,避免有歧义而导致判断结果有误。

 [root@yuji ~]# empty=              //设置一个空值变量
 [root@yuji ~]# echo $empty
 ​
 [root@yuji ~]# [ -z $empty ]       //测试变量是否为空值,为空
 [root@yuji ~]# echo $?
 0
 [root@yuji ~]# [ -n $empty ]       //变量没加引号,导致判断结果有误
 [root@yuji ~]# echo $?
 0
 [root@yuji ~]# [ -n "$empty" ]     //加了引号,判断结果正确
 [root@yuji ~]# echo $?
 1
复制代码

3字符串2.png

1.4 逻辑测试(短路运算)

逻辑测试用于连接多个测试条件,并返回整个表达式的值。 逻辑测试主要有"逻辑与","逻辑或","逻辑否"三种。

逻辑测试两种格式:

 格式一:[  表达式1  ]  逻辑运算符  [  表达式2  ] ...
 #例:[表达式1] && [表达式2]  等同于  [表达式1 -a 表达式2]  等同于  [[表达式1 && 表达式2]]
 #例:[表达式1] || [表达式2]  等同于  [表达式1 -o 表达式2]  等同于  [[表达式1 || 表达式2]]
 ​
 格式二:命令1  逻辑运算符  命令2  ... 
复制代码

逻辑运算符:

  • -a或&& :逻辑与,“而且”的意思,全真才为真。
  • -o或|| :逻辑或,“或者”的意思,一真即为真。
  • ! :逻辑否。

image-20220328120025087.png

1)短路与 &&

 CMD1 && CMD2 
 全真才为真,一假即为假。      
 第一个CMD1结果为真 ,第二个CMD2必须要参与运算,才能得到最终的结果。
 第一个CMD1结果为假 ,总的结果必定为假,因此不需要执行CMD2。
 同时满足命令1 和命令2 的要求,才会返回正确。
复制代码

2)短路或 ||

 CMD1 || CMD2
 一真即为真
 第一个CMD1结果为真,总的结果必定为真,因此不需要执行CMD2。
 第一个CMD1结果为假,第二个CMD2必须要参与运算,才能得到最终的结果。
复制代码

1.4.1 二元运算符

示例1:

 [root@yuji ~]# [ 1 -lt 2 ] && [ 3 -eq 4 ]    //测试1是否小于2,同时3是否等于4
 [root@yuji ~]# echo $?
 1                                            //结果为假
 [root@yuji ~]# [ 1 -lt 2 ] || [ 3 -eq 4 ]    //测试1是否小于2,或者3是否等于4
 [root@yuji ~]# echo $?
 0                                            //结果为真
复制代码

4逻辑测试1.png

示例2:

[表达式1] && [表达式2] 等同于 [表达式1 -a 表达式2] 等同于 [[表达式1 && 表达式2]]

 [root@localhost ~]# a=5
 [root@localhost ~]# [ $a -ne 1 ]&&[ $a != 2 ]
 [root@localhost ~]# echo $?
 0
 [root@localhost ~]# [ $a -ne 1 -a  $a != 2 ]
 [root@localhost ~]# echo $?
 0
 [root@localhost ~]# [[ $a -ne 1 &&  $a != 2 ]]
 [root@localhost ~]# echo $?
 0
复制代码

4逻辑测试2.png

1.4.2 三元运算符

1、java,C语言

  • 条件表达式?a:b
  • 当条件表达式为真时,取a值;当条件表达式为假时,取b值。

2、shell

  • [ 表达式 ] && a || b
  • 当表达式为真时,取a值;当表达式为假时,取b值。
 [ 表达式 ]  && a           //相当于 if语句单分支结构
 ​
 [ 表达式 ]  && a || b      //相当于 if语句双分支结构
 ​
 [ 表达式 ]  || b           //相当于 if [表达式];else
复制代码

案例应用:查看指定主机是否在线。

1)方法一:使用三元运算符

 [root@yuji ~]# vim ping001.sh       //使用三元运算符写脚本
 #!/bin/bash
 ping -c 3 -i 0.5 -w 3 $1 &>/dev/null && echo "$1 is up" || echo "$1 is down"
 ​
 [root@yuji ~]# bash ping001.sh 192.168.72.10      //执行脚本测试主机是否在线
 192.168.72.10 is up
 [root@yuji ~]# bash ping001.sh 192.168.72.22
 192.168.72.22 is down
复制代码

4逻辑测试三元运算符1.png

4逻辑测试三元运算符2.png

2)方法二:使用 if语句双分支结构

 [root@yuji ~]# vim ping002.sh               //使用if语句写脚本
 #!/bin/bash
 ping -c 3 -i 0.5 -w 3 $1 &>/dev/null 
 ​
 if [ $? -eq 0 ]
 then
     echo "$1 is online" 
 else    
     echo "$1 is offline"
 fi
 ​
 [root@yuji ~]# bash ping002.sh 192.168.72.10      //执行脚本测试主机是否在线
 192.168.72.10 is online
 [root@yuji ~]# bash ping002.sh 192.168.72.22
 192.168.72.22 is offline
复制代码

4逻辑测试三元运算符3.png

4逻辑测试三元运算符4.png

小贴士:

ping命令:

  • -c,发送包的个数。
  • -i,发送的间隔时间。
  • -w(小写),多少秒后停止ping操作。
  • -W(大写),以毫秒为单位设置ping的超时时间 。

2 if 语句

2.1 if 单分支结构

单分支语句格式:

 if  判断条件
 ​
 then  条件成立的分支代码
 ​
 fi                         //条件不成立直接结束
复制代码

if单分支.png

案例应用:

判断已用磁盘空间是否大于80%,如果是就报警提示。

 # 使用df命令可以查看磁盘空间使用和剩余情况
 [root@yuji ~]# df
 文件系统                   1K-块    已用     可用 已用% 挂载点
 /dev/mapper/centos-root 17811456 5063256 12748200   29% /
 devtmpfs                  483928       0   483928    0% /dev
 tmpfs                     499848       0   499848    0% /dev/shm
 tmpfs                     499848    7264   492584    2% /run
 tmpfs                     499848       0   499848    0% /sys/fs/cgroup
 /dev/sda1                1038336  182372   855964   18% /boot
 tmpfs                      99972      12    99960    1% /run/user/42
 tmpfs                      99972       0    99972    0% /run/user/0
 [root@yuji ~]# df |grep /dev/mapper/centos-root
 /dev/mapper/centos-root 17811456 5063288 12748168   29% /
 [root@yuji ~]# df |grep /dev/mapper/centos-root |awk '{print $5}'
 29%
 [root@yuji ~]# df |grep /dev/mapper/centos-root |awk '{print $5}' |awk -F '%' '{print $1}'
 29
 ​
 ​
 # 编写脚本
 [root@yuji ~]# vim diskused.sh
 #!/bin/bash
 # 判断已用磁盘空间是否大于80%,如果是则报警提示。
 ​
 used=$(df |grep /dev/mapper/centos-root |awk '{print $5}' |awk -F '%' '{print $1}')
 ​
 if [ $used -gt 80 ]
 then
    echo "警告!当前 /目录的磁盘使用率为 ${used}%"
 fi
复制代码

if单分支1.png

if单分支2.png

测试脚本:

可以先将判断条件修改为大于20%,测试脚本是否生效。

if单分支3.png

2.2 if 双分支结构

双分支语句格式:

 if 判断条件
 then
     条件成立的分支代码
 else
     条件不成立的分支代码
 fi
复制代码

if双分支.png

案例应用:

检测80端口是否在监听,如果是,则提示”网站服务已在运行“;如果否,则启动httpd服务。

 # 写脚本
 [root@yuji ~]# vim httpd80.sh
 #!/bin/bash
 # 检测80端口是否在监听,如果是,则提示”网站服务已在运行“;如果否,则启动httpd服务。
 ​
 netstat -ntap | grep :80 &> /dev/null
 ​
 if [ $? -eq 0 ]
 then
     echo "网站服务已在运行"
 else
     if rpm -q httpd &> /dev/null
     then
         echo "正在启动httpd服务"
         systemctl start httpd
     else
         echo "正在安装并启动httpd服务"
         yum install httpd -y &> /dev/null
         systemctl start httpd &> /dev/null
    fi
 fi
         
 # 执行脚本
 [root@yuji ~]# bash httpd80.sh       //检测到未安装httpd服务,自动安装
 正在安装并启动httpd服务
 [root@yuji ~]# rpm -q httpd          //查看软件包是否安装,已安装
 httpd-2.4.6-97.el7.centos.5.x86_64
 [root@yuji ~]# bash httpd80.sh      //httpd服务已运行
 网站服务已在运行
复制代码

if双分支1.png

if双分支2.png

2.3 if 多分支结构

多分支语句格式:

 if 判断条件1
 then
     条件1为真的分支代码
 ​
 elif 判断条件2
 then
     条件2为真的分支代码
 ​
 elif 判断条件3
 then
     条件3为真的分支代码
 ​
 ...
 else
     以上条件都为假的分支代码
 ​
 fi
复制代码

if多分支.png

案例应用:

查看当前时间点,根据不同时间范围输出不同问候语。

  • 6点-10点,输出早上好;
  • 11点-13点,输出中午好;
  • 14点-18点,输出下午好;
  • 19点-22点, 输出晚上好;
  • 其他时间,休息时间。
 [root@yuji ~]# date          //查看当前详细时间
 2022年 04月 10日 星期日 15:01:25 CST
 [root@yuji ~]# date +%H      //查看当前时间在哪个小时(范围0-23)
 15
 [root@yuji ~]# vim date.sh     //写脚本
 #!/bin/bash
 # 查看当前时间点,根据不同时间范围输出不同问候语。
 ​
 h=$(date +%H)
 ​
 if [ $h -ge 6 -a $h -le 10 ]
 then
     echo "早上好"
 elif [ $h -ge 11 -a $h -le 13 ]
 then
     echo "中午好"
 elif [ $h -ge 14 -a $h -le 18 ]
 then
     echo "下午好"
 elif [ $h -ge 19 -a $h -le 22 ]
 then
     echo "晚上好"
 else
     echo "休息时间"
 fi
 ​
 [root@yuji ~]# bash date.sh      //执行脚本
 下午好
复制代码

if多分支1.png

if多分支2.png

3 case语句

格式:

 case 变量引用 in
 模式1)
     命令分支1
     ;;
 模式2)
     命令分支2
     ;;
 ...
 *)
     默认命令分支
 esac
复制代码

注意事项:

 case支持glob风格的通配符:
 * :任意长度任意字符
 ? :任意单个字符
 [0-9] :指定范围内的任意单个字符
 | :“或者”的意思,如: a|b
复制代码

案例应用:

提示用户输入分数(0-100),判断分数范围,分出优秀、良好、及格、不及格四档。如果用户输入的分数值不在0-100之间,则提示重新输入。

 # 写脚本
 [root@localhost ~]# vim gradediv.sh
 #!/bin/bash
 #提示用户输入分数(0-100),判断分数范围,分出优秀、良好、及格、不及格四档。
 #如果用户输入的分数值不在0-100之间,则提示重新输入。
 ​
 read -p "请输入你的分数(0-100):" grade
 ​
 case $grade in
 100)
     echo "你很优秀"
     ;;
 [89][0-9])
     echo "你表现良好"
     ;;
 [67][0-9])
     echo "你及格了"
     ;;
 [0-9]|[1-5][0-9])
     echo "你不及格"
     ;;
 *)
     echo "输入有误,请重新输入0-100"
     bash $0
 esac
 ​
 # 执行脚本
 [root@localhost ~]# bash gradediv.sh
 请输入你的分数(0-100):0
 你不及格
 [root@localhost ~]# bash gradediv.sh
 请输入你的分数(0-100):33
 你不及格
 [root@localhost ~]# bash gradediv.sh
 请输入你的分数(0-100):67
 你及格了
 [root@localhost ~]# bash gradediv.sh
 请输入你的分数(0-100):92
 你表现良好
 [root@localhost ~]# bash gradediv.sh
 请输入你的分数(0-100):100
 你很优秀
 [root@localhost ~]# bash gradediv.sh
 请输入你的分数(0-100):300
 输入有误,请重新输入0-100
 请输入你的分数(0-100):85
 你表现良好
复制代码

case1.png

case2.png

4 易错总结

  1. 测试文件是否存在时,建议使用-e。如:[ -e /etc/passwd ]
  2. root用户的读取和写入权限,不受文件的基本权限控制;但执行权限受文件基本权限控制。
  3. 测试软链接时,目录名称后面不要加"/",不然会被判断为目录文件。如:[ -L /bin ]
  4. 使用-z或-n测试变量时,变量需要加引号,避免有歧义而导致判断结果有误。如:[ -n "$empty" ]