Shell 编程之条件语句

条件测试操作

要使 Shell脚本程序具备一定的“智能”,面临的第一个问题就是如何区分不同的情况以 确定执行何种操作。例如,当磁盘使用率超过95%时,发送告警信息;当备份目录不存在时,能够自动创建;当源码编译程序时,若配置失败则不再继续安装等。

Shell 环境根据命令执行后的返回状态值($?) 来判断是否执行成功,当返回值为0时 表示成功,否则(非0值)表示失败或异常。使用专门的测试工具——test 命令,可以对特定条件进行测试,并根据返回值来判断条件是否成立(返回值为0表示条件成立)。

使用 test 测试命令时,包括以下两种形式。

test 条件表达式 [ 条件表达式 ]

这两种方式的作用完全相同,但通常后一种形式更为常用,也更贴近编程习惯。需要注意的是,方括号“[”或“]”与条件表达式之间需要至少一个空格进行分隔。

根据需要测试的条件类别不同,条件表达式也不同。比较常用的条件操作包括文件测试、整数值比较、字符串比较,以及针对多个条件的逻辑测试,下面分别进行介绍。

文件测试

文件测试指的是根据给定的路径名称,判断对应的是文件还是目录,或者判断文件是否 可读、可写、可执行等。文件测试的常见操作选项如下,使用时将测试对象放在操作选项之后即可。

>   -d: 测试是否为目录 ( Directory)。

>   -e: 测试目录或文件是否存在 ( Exist)。

>  -f:   测试是否为文件 ( File )。

> -r:      测试当前用户是否有权限读取( Read)  。

>    -w: 测试当前用户是否有权限写入 (Write)。

>   -x:   测试是否设置有可执行 ( Excute)    权限。

执行条件测试操作以后,通过预定义变量$? 可以获得测试命令的返回状态值,从而判 断该条件是否成立。例如,执行以下操作可测试目录/media/是否存在,如果返回值$?为 0 ,

表示存在此目录,否则表示不存在或者虽然存在但不是目录。

[root@localhost~]#[-d /media/] //查看前一命令的返回值 
[root@localhost ~]# echo $? //返回0表示条件成立 0

若测试的条件不成立,则测试操作的返回值将不为0(通常为1)。

通过查看变量$? 的值可以判断前一步的条件测试结果,但是操作比较烦琐,输出结果 也并不是很直观。为了更直观地查看测试结果,可以结合命令分隔符“&& ”和echo   命令一起 使用,当条件成立时直接输出“YES”。 其 中 ,“&&”符号表示“而且”的关系,只有当前面的命令执行成功后才会执行后面的命令,否则后面的命令将会被忽略。

整数值比较

整数值比较指的是根据给定的两个整数值,判断第一个数与第二个数的关系,如是否大于、等于、小于第二个数。整数值比较的常用操作选项如下,使用时将操作选项放在要比较的两个整数之间。

>   -eq:   第一个数等于 (Equal ) 第二个数。

>   -ne: 第一个数不等于 (Not  Equal)第二个数。

>   -gt: 第一个数大于 ( Greater Than) 第二个数。

>   -It:第一个数小于 ( Lesser Than) 第二个数。

>   -le:   第一个数小于或等于 ( Lesser  or  Equal) 第二个数。

>   -ge: 第一个数大于或等于 ( Greater or Equal) 第二个数。

整数值比较在 Shell 脚本编写中的应用较多。例如,用来判断已登录用户数量、开启进 程数、磁盘使用率是否超标,以及软件版本号是否符合要求等。实际使用时,往往会通过变量引用、命令替换等方式来获取一个数值。

字符串比较

字符串比较通常用来检查用户输入、系统环境等是否满足条件,在提供交互式操作的 Shell  脚本中,也可用来判断用户输入的位置参数是否符合要求。字符串比较的常用操作选项如下。

>   =:第一个字符串与第二个字符串相同。

>   !=:第一个字符串与第二个字符串不相同,其中“!”符号表示取反。

>   -z:   检查字符串是否为空 (Zero),     对于未定义或赋予空值的变量将视为空串。

逻辑测试

逻辑测试指的是判断两个或多个条件之间的依赖关系。当系统任务取决于多个不同的条  件时,根据这些条件是否同时成立或者只要有其中一个成立等情况,需要有一个测试的过程。

常用的逻辑测试操作如下,使用时放在不同的测试语句或命令之间。

>  &&: 逻辑与,表示“而且”,只有当前后两个条件都成立时,整个测试命令的返回值才为0(结果成立)。使用test命令测试时,“&&”可改为“-a”。

>  II:   逻辑或,表示“或者”,只要前后两个条件中有一个成立,整个测试命令的返回值即为0 (结果成立)。使用test  命令测试时,“I” 可改为“-o”。

>  !:逻辑否,表示“不”,只有当指定的条件不成立时,整个测试命令的返回值才为0(结果成立)。

在上述逻辑测试的操作选项中,“&&”和“I” 通常也用于间隔不同的命令操作,其作用是相似的。

if 条件语句

通过上一节中的条件测试操作,实际上使用“&&”和“II”逻辑测试已经可以完成简单的判 断并执行相应的操作,但是当需要选择执行的命令语句较多时,这种方式将使执行代码显得很复杂,不好理解。而使用专用的if 条件语句,可以更好地整理脚本结构,使得层次分明,清晰易懂。

if 语句的结构

在Shell  脚本应用中, if 语句是最为常用的一种流程控制方式,用来根据特定的条件测 试结果,分别执行不同的操作(如果……那么……)。根据不同的复杂程度,if   语句的选择结构可以分为三种基本类型,适用于不同的应用场合。

1. 单 分 支 if 语 句

f 语句的“分支”指的是不同测试结果所对应的执行语句(一条或多条)。对于单分支的选择结构,只有在“条件成立”时才会执行相应的代码,否则不执行任何操作。单分支if 语句的语法格式如下所示。

if 条件测试操作 then 命令序列 fi

在上述语句结构中,条件测试操作既可以是“[条件表达式]”语句,也可以是其他可执行的命令语句;命令序列指的是一条或多条可执行的命令行,也包括嵌套使用的if 语句或其他流程控制语句。

单分支if语句的执行流程:首先判断条件测试操作的结果,如果返回值为0,表示条件成立,执行then后面的命令序列,一直到遇见fi结束判断为止,继续执行其他脚本代码;如果返回值不为0,则忽略 then 后面的命令序列,直接跳至fi 行以后执行其他脚本代码,如图2.1 所示。

2.双分支if 语句

对于双分支的选择结构,要求针对“条件成立”“条件不成立”两种情况分别执行不同的操作。双分支if 语句的语法格式如下所示。

if 条件测试操作 
then 
命令序列1 
else 
命令序列2

双分支if 语句的执行流程:首先判断条件测试操作的结果,如果条件成立,则执行then  后面的命令序列1,忽略else  及后面的命令序列2,直到遇见fi 结束判断;如果条件不成立,则忽略then及后面的命令序列1,直接跳至else后面的命令序列2并执行,直到遇见fi结束判断。

3.多分支if 语句

由于if语句可以根据测试结果的成立、不成立分别执行操作,所以能够嵌套使用,进行多 次判断。例如,首先判断某学生的得分是否及格,若及格则再次判断是否高于90 分等。多分支 if 语句的语法格式如下。

if 条件测试操作1 
then 
命令序列1 
elif 条件测试操作2 
then 
命令序列2 
else 
命令序列3 
fi

上述语句结构中只嵌套了一个 elif 语句作为示例,实际上可以嵌套多个。if 语句的嵌套在编写 Shell 脚本时并不常用,因为多重嵌套容易使程序结构变得复杂。当确实需要使用多分支的程序结构时,采用下一节的case   语句更加方便。

多分支if语句的执行流程:首先判断条件测试操作1的结果,如果条件1成立,则执 行命令序列1,然后跳至fi结束判断;如果条件1不成立,则继续判断条件测试操作2 的 结 果,如果条件2 成立,则执行命令序列2, 然后跳至fi结束判断……如果所有的条件都不满

足,则执行 else 后面的命令序列 n,   直到遇见fi 结束判断,如图2.3 所示。

 case 分支语句

case   语句可以使脚本程序的结构更加清晰、层次分明,本节就来学习case    语句的语法结构及应用。

case语句的结构

case   语句主要适用于以下情况:某个变量存在多种取值,需要对其中的每一种取值分 别执行不同的命令序列。这种情况与多分支的if语句非常相似,只不过if语句需要判断多个不同的条件,而 case 语句只是判断一个变量的不同取值。

case   分支语句的语法结构如下所示。

case 变量值 in 
模式1) 
命令序列1 
;; 
模式2) 
命令序列 2 
;; 
*) 
默认命令序列 
esac

在上述语句结构中,关键字case  后面跟的是“变量值”,即“$变量名”。整个分支结构包 括在 case…esac 之间,中间的模式1、模式2、 ……、*对应为变量的不同取值(程序期望的取值),其中*作为通配符,可匹配任意值。

case   语句的执行流程:首先使用“变量值”与模式1进行比较,若取值相同则执行模式1后的命令序列,直到遇见双分号“;”后跳转至 esac,    表示结束分支;若与模式1不相匹配,则继续与模式2 进行比较,若取值相同则执行模式2 后的命令序列,直到遇见双分号“;;”后   跳转至esac, 表示结束分支……依此类推,若找不到任何匹配的值,则执行默认模式“*)”后的命令序列,直到遇见esac   后结束分支,如图 3.3所示。

使用case 分支语句时,有几个值得注意的特点如下所述。

>   case 行尾必须为单词“in”,每一模式必须以右括号“)”结束。

>   双分号“;;”表示命令序列的结束。

>  模式字符串中,可以用方括号表示一个连续的范围,如“[0-9]”;还可以用竖杠符号“T”表示或,如“A|B”。

>   最后的“*)”表示默认模式,其中的*相当于通配符。

在 Linux  系统中,源码软件包编译安装后提供的服务控制脚本使用了 case    分支语句;

也有 一 些源码包没有提供服务控制脚本,编译安装后可参照上例自行编写服务控制脚本。平时控制各种系统服务时,提供的 start 、stop 、restart 等位置参数,正是由 case  语句结构来识别并完成相应操作的。有兴趣的同学可自行查阅这些脚本内容。

若要将 myprog 服务交给systemd来管理,还需要在/lib/systemd/system目录下添加相应的myprog.service 配置文件。

  • 24
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值