一、文件测试
Shell 环境根据命令执行后的返回状态值($?)来判断是否执行成功,当返回值为 0 时表示成功,否则(非 0 值)表示失败或异常。
但是我们还可以使用专门的测试工具——test 命令,可以对特定条件进行测试,并根据返回值来判断条件是否成立(返回值为 0 表示条件成立)。
注意中括号和表达式中间需要隔一个空格
语法格式:
格式1 test 条件表达式
格式2 [ 条件表达式 ]格式3 [[ 条件表达式 ]]
文件测试指的是根据给定的路径名称,判断对应的是文件还是目录,或者判断文件是否可读、可写、可执行等。
文件测试的常见操作选项如下,使用时将测试对象放在操作选项之后即可。
- -d:测试是否为目录(Directory)
- -e:测试目录或文件是否存在(Exist)
- -f:测试是否为文件(File)
- -r:测试当前用户是否有权限读取(Read)
- -w:测试当前用户是否有权限写入(Write)
- -x:测试是否设置有可执行(Excute)权限
- -b:测试是否为设备文件
- -c:测试是否为字符设备文件
- -s:测试存在且文件大小为空
- -L:测试是否为链接文件
执行条件测试操作以后,通过预定义变量$?可以获得测试命令的返回状态值,从而判断该条件是否成立。
例如,执行以下操作可测试目录/media/是否存在,如果返回值$?为 0, 表示存在此目录,否则表示不存在或者虽然存在但不是目录
二、整数值比较
整数值比较指的是根据给定的两个整数值,判断第一个数与第二个数的关系,如是否大于、等于、小于第二个数。
整数值比较的常用操作选项如下,使用时将操作选项放在要 比较的两个整数之间。
常用测试操作符
- -eq:第一个数等于(Equal)第二个数。
- -ne:第一个数不等于(Not Equal)第二个数。
- -gt: 第一个数大于(Greater Than)第二个数。
- -lt: 第一个数小于(Lesser Than)第二个数。
- -le: 第一个数小于或等于(Lesser or Equal)第二个数。
- -ge:第一个数大于或等于(Greater or Equal)第二个数
三、字符串比较
字符串比较通常用来检查用户输入、系统环境等是否满足条件,在提供交互式操作的Shell脚本中,也可用来判断用户输入的位置参数是否符合要求。字符串比较的常用操作选项如下。
- =:第一个字符串与第二个字符串相同。
- !=:第一个字符串与第二个字符串不相同,其中“!”符号表示取反。
- -z:检查字符串是否为空(Zero),对于未定义或赋予空值的变量将视为空串
四、逻辑测试
逻辑测试指的是判断两个或多个条件之间的依赖关系。
当系统任务取决于多个不同的条件时,根据这些条件是否同时成立或者只要有其中一个成立等情况,需要有一个测试的过程。
常用的逻辑测试操作如下,使用时放在不同的测试语句或命令之间。
- &&:逻辑与,表示“而且”,只有当前后两个条件都成立时,整个测试命令的返回值 才为 0(结果成立)。使用 test命令测试时,“&&”可改为“-a”。
- ||:逻辑或,表示“或者”,只要前后两个条件中有一个成立,整个测试命令的返回值即为 0(结果成立)。使用 test 命令测试时,“||”可改为“-o”。
- !:逻辑否,表示“不”,只有当指定的条件不成立时,整个测试命令的返回值才为 0(结果成立)。
在上述逻辑测试的操作选项中,“&&”和“||”通常也用于间隔不同的命令操作,其作用是相似的。实际上此前已经接触过“&&”操作的应用,如“make && make install”的编译安装操作。
4.1、实验
若要判断当前 Linux 系统的内核版本是否大于 2.11,可以执行以下操作。其中, 内核版本号通过 uname 和 awk 命令获得
[ $version -gt 2 ] || ( [ $version -eq 2 ] && [ $num -gt 11 ] ) && echo "版本正确大于2.6 "
版本正确大于2.11
五、if语句的结构
注意: 【】为[ ] 为了更清楚。
单个括号(【】):必须要有空格 两个字符或数字之间的比较左右必须有空格
条件判断:test一种形式,【】中间只能使用 = 和 != 比较字符串 ,如果使用 < 或 >需要进行转义\.然后中间如果比较数字可以用 -lt 等符号。
两个括号(【【】】):可用于处理逻辑命令,也可以处理字符串是否相等,且使用 < 、>不用转义符。
5.1 单分支 if 语句
实际上使用“&&”和“||”逻辑测试已经可以完成简单的判断并执行相应的操作,但是当需要选择执行的命令语句较多时,这种方式将使执行代码显得很复杂,不好理解。
而使用专用的 if 条件语句,可以更好地整理脚本结构,使得层次分明, 清晰易懂。
单分支 if 语句:对于单分支的选择结构,只有在“条件成立”时才会执行相应的代码,否则不执行任何操作。
语法格式:
注意后面有个结尾,开头结尾要凑成一对否则会报语法错误
if 条件测试操作
then
命令序列
fi
或
if 条件测试操作;then
命令序列
fi
5.1.1、一些实验
例1 判断目录是否存在
例2判断挂载点,如果不存在就自动创建
例3 判断输入结尾是否.sh
5.2、双分支 if 语句
双分支 if 语句只是在单分支的基础上针对“条件不成立”的情况执行另一种操作,而不是 “坐视不管”地不执行任何操作
例1 判断目标主机是否存活,存活就打印is up,如果不存活就打印is down
-c ping的次数
-i 每次多久ping一次,单位是秒
-W 反馈结果的时间,如果不通时可加快反馈时间,单位是秒
例2.判断输入的用户名是否存在,存在提示已存在,不存在则创建用户并设置密码
例3.判断当前登陆用户是否是管理员
例4.检查目录是否存在
或者
、
例5,启动http服务
#!/bin/bash
netstat -natup |grep ":80" &> /dev/null
if [ $? -eq 0 ];then
echo "网站已经运行了"
elif
systemctl start httpd
[ $? -eq 0 ];then
echo "启动中"
echo "启动成功"
else
echo "你好像都没有包,正在下载中"
yum install -y httpd > /dev/null
systemctl start httpd
echo "启动成功"
fi
~
5.3、多分支if语句
与单分支、双分支 if 语句相比,多分支 if 语句的结构能够根据多个互斥的条件分别执行不同的操作
例1.根据输入的考试分数不同来区分优秀、合格、不合格三挡
还可以 elif [ $score -ge 60 -a $score -le 69 ]
例2.判断文件类别
5.4、嵌套if语句
就是确认大方向再细分内容。
例1.判断httpd服务有没有启动
if 判断是否启动
if 如果启动------输出已启动
如果没启动----判断是否安装---如果安装---启动
如果没安装----安装---如果安装成功---启动
如果安装不成功-----报错
#!/bin/bash
netstat -anptul |grep ":80" &>/dev/null
if [ $? -ne 0 ];then
if [ "$(rpm -q httpd)" != 0 ] ;then
yum install -y httpd &>/dev/null
systemctl start httpd
echo "1"
else
systemctl start httpd
echo "2"
fi
else
echo "已经运行了"
fi
或者 [ "$(rpm -q httpd)" == "未安装软件包 httpd " ] && echo "ok"
例2
判断是否有用户 判断一个用户是否有家目录,有打印正常,没有询问是否需要删除,如果是就删除
判断系统有无此用户----有---判断有无家目录----有----输出正常的用户
无-----询问是否删除此用户----是----删除
否----退出脚本
无----提示没有此用户
系统的用户:
有用户也有家目录
有用户但是没家目录
无用户但是有家目录
#!/bin/bash
read -p "请输入用户名" name
if (grep $name /etc/passwd) &>/dev/null;then
if [ -d /home/$name ];then
echo "状态正常"
else
read -p "该用用户没有家目录,是否删除用户,请输入yes/no" ack
if [ $ack == yse ];then
echo "正在删除用户"
userdel -r $name &> /dev/null
elif [$ask ==no ];then
exit
fi
fi
else
echo "用户不存在"
fi
六、case语句
case 语句可以使脚本程序的结构更加清晰、层次分明,常用于服务的启动、重启、停止的脚本,有的服务不提供这种控制脚本,需要用case语句编写。
case 语句主要适用于以下情况:某个变量存在多种取值,需要对其中的每一种取值分别执行不同的命令序列。这种情况与多分支的 if 语句非常相似,只不过 if 语句需要判断多个 不同的条件,而 case 语句只是判断一个变量的不同取值
格式
case 变量值 in
模式 1)
命令序列 1
;;
模式 2)
命令序列 1
;;
* )
默认命令序列
esac
case 行尾必须为单词“in”,每一模式必须以右括号“)”结束。
双分号“;;”表示命令序列的结束。 模式字符串中,可以用方括号表示一个连续的范围,如“[0-9]”;还可以用竖杠符号 “|”表示或,如“A|B”。 最后的“*)”表示默认模式,其中的*相当于通配符。
#!/bin/bash
# 提示用户输入一个数字
echo "请输入一个数字(1-3):"
read number
# 使用case语句根据输入的数字执行不同的操作
case $number in
1)
echo "你输入了数字 1"
;;
2)
echo "你输入了数字 2"
;;
3)
echo "你输入了数字 3"
;;
*)
echo "无效输入,请输入1、2或3"
;;
esac
典型案例:检查用户输入的字符类型
#!/bin/bash
read -p "请输入一个字符,并且Enter确认" KEY
case $KEY in
[a-z]| [A-Z])
echo "你输入的是字母"
;;
[0-9])
echo "你输入的是数字"
;;
*)
echo "你输入的是其他类型特殊字符"
esac
分数案例
#!/bin/bash
read -p "请输入一个分数,并且Enter确认" score
[[ $score -eq 100 ]] && a="满分"
[ $score -lt 100 ] && [ $score -gt 79 ] && a=b
[ $score -lt 80 -a $score -gt 60 ] && a="良好"
case $a in
满分)
echo "抄0遍"
;;
b)
echo "抄10遍"
;;
良好)
echo "抄20遍"
;;
*)
echo "你惨了"
;;
esac
~