一.条件测试
1.文件测试与整数测试
2.字符串测试与逻辑测试
二.if语句
1.if单分支语句
2.if双分支语句
3.if多分支语句
三.case分支语句
一.条件测试
(一).条件测试操作
1.test命令
测试表达式是否成立,若成立返回0,否则返回其他数值
格式1: test条件表达式
格式2:【条件表达式】
要使shell脚本程序具备一定的"智能",面临的第一个问题就是如何区分不同的情况以确定执行何种操作。例如,当磁盘使用率超过95%时,发送告警信息;当备份目录不存在时,能够自动创建:当源码编译程序时,若配置失败则不再继续安装等Shell环境根据命令执行后的返回状态值($?〉来判断是否执行成功,当返回值为o时表示成功,否则(非О值)表示失败或异常。
使用专门的测试工具—test命令,可以对特定条件进行测试,并根据返回值来判断条件是否成立(返回值为0表示条件成立)。
语法格式:
格式l test条件表达式
格式2〔条件表达式]#注意中括号和表达式中间需要隔一个空格或
[[条件表达式]]
2.文件测试
[操作符文件或目录]
常用的测试操作符
文件测试指的是根据给定的路径名称,判断对应的是文件还是目录,或者判断文件是否
可读、可写、可执行等。文件测试的常见操作选项如下,使用时将测试对象放在操作选项之后即可。
-d:测试是否为目录(Directory) 。
-e:测试目录或文件是否存在(Exist ) 。
-f:测试是否为文件( File ) 。
-r:测试当前用户是否有权限读取(Read)。
-w:测试当前用户是否有权限写入(write ) 。
-x:测试是否设置有可执行(Excute)权限。
-b:测试是否为设备文件
-c:测试是否为字符设备文件
-s:测试存在且文件大小为空
-L:测试是否为链接文件
执行条件测试操作以后,通过预定义变量$?可以获得测试命令的返回状态值,从而判断该条件是否成立。例如,执行以下操作可测试目录/media/是否存在,如果返回值$?为 0,
表示存在此目录,否则表示不存在或者虽然存在但不是目录
例1
test -d letcl sysconfig/
test -f letc/ sysconfig/
test -e letcl sysconfig/
例2
[ -f / home / humajun/ ]
[ -d / home / humajun/ ] & & echo "YES"
[ ! -e /opt/ kgc ] & & mkdir /opt/ kgc
3.整数值比较
[整数1操作符整数2]
常用的测试操作符
-eq:等于(Equal)
-ne:不等于(Not Equal)
-gt:大于(Greater Than)
-lt:小于(Lesser Than)
-le:小于或等于(Lesser or Equal)
-ge:大于或等于(Greater or Equal)
1.2整数值比较
整数值比较指的是根据给定的两个整数值,判断第一个数与第二个数的关系,如是否大于、等于、小于第二个数。整数值比较的常用操作选项如下,使用时将操作选项放在要
比较的两个整数之间。
常用测试操作符
-eq:第一个数等于(Equal)第二个数。
-ne:第一个数不等于(Not Equal)第二个数。-gt:第一个数大于(Greater Than)第二个数。-lt:第一个数小于(Lesser Than)第二个数。
-le:第一个数小于或等于(Lesser or Equal)第二个数。-ge:第一个数大于或等于( Greater or Equal)第二个数
例1
who lwc -l
[ $ (who | wc -l) -le 5 ] &&echo "用户太少"
[ $(who / wc -l) -ge 10 ] && echo "> = 10. "
例2
FreecC=$ (free -m l grep "Mem: " | awk '{print $6} ')
[$Freecc -lt 4096 ]&& echo ${FreecC}MB
mem_size=$ (free -m | grep "Mem: " | awk '{print $4} ')
[$mem_size -ge 1024 ]&& echo
"由于192.168.10.17服务器系统内存资源不足,请尽快处理故障."
1.3字符串比较
字符串比较通常用来检查用户输入、系统环境等是否满足条件,在提供交互式操作的shell脚本中,也可用来判断用户输入的位置参数是否符合要求。字符串比较的常用操作选项如下。
-:第一个宁符串与第二个字符串相同。
!=:第一个字符串与第二个字符串不相同,其中"!"符号表示取反。
-z检查字符串是否为空(zero),对于未定义或赋予空值的变量将视为空串。
例1
echo $LANGzh_CN.UTF-8
[ $LANG!="en. us" ]& & echo "Not en.us"
例2
read -p "是否覆覆盖现有文件( yes/ no ) ?"
ACK是否覆覆盖现有文件
( yes/no ) ?yes
[$ACK = "yes" ]&& echo"覆盖"
1.4 逻辑测试
逻辑测试指的是判断两个或多个条件之间的依赖关系。当系统任务取决于多个不同的条件时,根据这些
条件是否同时成立或者只要有其中一个成立等情况,需要有一个测试的过程。
常用的逻辑测试操作如”下,使用时放在不同的测试语句或命令之间。
&&:逻辑与,表示"而且”,只有当前后两个条件都成立时,整个测试命令的返回值才 为
0 (结果成立)。使用test命令测试时,“&&”可改为"-a”。
●11:逻辑或,表示"或者”,只要前后两个条件中有一“个成立,整个测试命令的返回值即为0 (结果成立)。使用test命令测试时,“11”可改为"-o”。
●!:逻辑否,表示"不”,只有当指定的条件不成立时,整个测试命令的返回值才为0 (结果成立)。
例1
[ -d /etc ] && [ -r /etc ] && echo "You can open it"
[ -d /etc ] || [ -d /home ] && echo "ok"
在上述逻辑测试的操作选项中,"&&" 和“11"通常也用于间隔不同的命令操作,其作用是相似的。实际上此前已经接触过“&&"操作的应用,如“make && make install"的编译 安装操作。
例如,若要判断当前Linux 系统的内核版本是否大于3.4, 可以执行以下操作。其中,
内核版本号通过uname和awk命令获得
例2
uname - r
//查 看内核版本信息3.10.0-514.e17.x86_ 64 I
Mnum=$ (uname -r I awk -F. ' {print $1}') //取主版本号
Snum=$ (uname -r | awk -F. '{print $2}') //取次版本号
[ $Mnum-ge3 ] && [ $Snum-gt4 ] && echo"符合要求"
二.if语句单分支
(一).单分支if语句
实际上使用"&&"和“11"逻辑测试已经可以完成简单的判断并执行相应的操作,但是当需要选择执行的命
令语句较多时,这种方式将使执行代码显得很复杂,不好理解。而使用专用的if
条件语句,可以更好地整理脚本结构,使得层次分明,
清晰易懂
(二).单分支if
语句:对于单分支的选择结构,只有在“条件成立”时才会执行相应的代码,否则不执行任何操作。
语法格式:
1)
if条件测试操作
then
命令序列
fi
#注意后面有个结尾,开头结尾要凑成一对否则会报语法错误
2)
if条件测试操作; then
命令序列
fi
if[3 -gt 2]; then .
echo "ok""
fi
[3-gt2] & &echo"ok"
例1判断目录是否存在
#! /bin/bash
if ls /mnt
then
echo "it's ok"
fi
例2
判断挂载点,如果不存在就自动创建
#! /bin/bash
MOUNT_ DIR= "/media/ cdrom/”
#2.判断没有这个目录就创建一个目录
if[ !-d$MOUNTDIR ]
then
mkdir -P $MOUNT_DIR
fi
例3判断输入结尾是否.sh
#!/bin/bash
read -p "请输入文件名:”file
If [ [ $file == *.sh ] ]; then
echo "这是一个shell脚木"
fi
双分支if语句只是在单分支的基础上针对"条件不成立”的情况执行另一种操作,而不是
“坐视不管”地不执行任何操作
例1判断目标主机是否存活,存活就打印is up, 如果不存活就打印is down
C ping的次数
-1每次多久ping-一次,单位是秒
-W反馈结果的时间,如果不通时可加快反馈时间,单位是秒
#! /bin/bash
ip=192.168.10.10
ping -c 2 -i 0.2 -W 3 $ip &> /dev/null
if[$?-eq0];then
echo "$ip is up "
else
echo "$ip is down"
fi
判断以. sh结尾的脚本是一个shell脚本(双方括号用法:可以模式匹配)
[abc==abc]
[[ abc == abc* ]]
例4
#! /bin/bash .
read -p "检查目录是否存在,请输入目录:”
aaa
if ls $aaa > /dev/null
then
echo”目录存在"
else
echo "请输入正确路径"
fi
例5
#! /bin/bash
netstat -natup I grep ":80" &> /dev/nu1l
if[$?-eq0];then
echo "网站服务已经运行!”
else
echo "启动httpd服务" .
yum install -y httpd > /dev/null
systemctl start httpd
fi
(三).嵌套if语句
例1.判断httpd服务有没有启动
判断是否启动
如果启动------输出己启动
如果没启动----判断是否安装---如果安装---启动
如果没安装----安装---如果安装成功---启动
如果安装不成功--—--报错
ps aux l grep httpd l grep -v grep
if [ $? -ne o ] ;then
if [ "$ (rpm -q httpd) " == "未安装软件包httpd " ];then
yum -y install httpd
systemctl start httpd
else
systemctl start httpd
fi
else
echo "httpd is running"
fi
例2.判断一个用户是否有家目录,有打印正常,没有询问是否需要删除,如果是就删除判断系统有无比用户----有---判断有无家目录----有----输出正常的用户
无-----询问是否删除此用户----是----删除
否----退出脚本
无----提示没有此用户
例1系统的用户:
有用户也有家目录
有用户但是没家目录
无用户但是有家目录
read -p "请输入用户名:"user
if grep $user fetc/passwd &> / dev/null;then
if[ -d f home/ suser ] ;then
echo"该用户状态正常"
else
read -p "该用户没有家目录,是否删除该用户,请输入[yes/no]: " ask
if [ $ask == yes ] ; then
echo "正在删除用户..."
userdel $user_6>_ldev/null; sleep 2
echo"该用户已副除"
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".
最后的“*)”表示默认模式,其中的*相当于通配符。
案例:
今天吃什么? - -周七天不重样
case语句的执行流程:首先使用"变量值"与模式1进行比较,若取值相同则执行模式1
后的命令序列,直到遇见双分号“;;“后跳转至esac, 表示结束分支:若与模式1不相匹配, 则继续与模式2
进行比较,若取值相同则执行模式2后的命令序列,直到遇见双分号“;;”后跳转至
esac,表示结束分支... .依此类推,若找不到任何匹配的值,则执行默认模式“*)”后的命令序列,直到遇见esac后结束分支
典型案例:检查用户输入的字符类型
#!/bin/bash
read -P "请输入一个字符,并按Enter键确认: " KEY
case "SKEY" ir
[a-z]1[A-Z])
echo "你输入的是字母”
;
[0-9])
echo "你输入的是数字"
::
*)
echo "你输入的是空格,*,_ ,等特殊字符"
esac
#!/bin/bash
read -P "请输入您的分数(0-100) :”score
[[ Sscore -ge 85 6& Sscore -le 100 ]] && a="great"
[[ $score -ge 70 &6 sscore -1t 85 ]] &6 a="standard"
[[ Sscore -ge 0 66 $score -1t 70 ]] 66 a="false"
case $a in
great)
echo "Sscore分,优秀!"
;;
standard)
echo "sscore 分,合格!"
;;
false )
echo "sscore 分,不合格!"
;;
*)
echo "输入有误!"
esac
第一种方法
#!/bin/bash
read -P“请输入你的分数:”score
[ Sscore -eq 100 ] && a=0
[ Sscore -ge 90 ] 66[ $score -1t 100 ] 66 a=10
[ Sscore -ge 70 -a $score -le 89 ] 66 a=20
[ Sscore -ge 60 -a $score -le 69 ] && a=30
[ $score -ge 0 -a $score -1t 60 ] && a=40
case $a in
echo "秀儿!”
;;
10)
echo "sscore 分,抄10遍!"
;;
20)
echo "Sscore 分,抄20遍!”
;;
30)
echo "sscore 分,抄30遍!"
;;
40 )
echo "sscore 分,全抄30遍!"
;;
*)
echo
”输入有误!”
esac
第二种方法
read -P”请输入你的分数:”score
case Sscore in
100 )
echo "秀儿!”
;;
9(0-9])
echo "Sscore 分,抄10遍!"
;;
[78](0-9])
echo "sscore 分,抄20遍!"
;;
6[0-9])
echo "Sscore 分,抄30遍!"
;;
[0-9]1[1-5][0-9] )
echo "Sscore 分,全抄30遍!"
;;
*)
echo”输入有误!”
esac
编写apache 启动服务脚本
#!/bin/bash
case $1 in
start)
/usr/bin/systemctl $1 httpd
/usr/bin/ps aux I grep httpd
echo "httpd start ”
;;
stop)
/usr/bin/systemct1 $1 httpd
/usr/bin/ps aux I grep httpd
echo "httpd stop"
;;
restart)
echo "httpd STOP..."
/usr/bin/ps aux I grep httpd
/usr/bin/systemct1 $1 httpd
echo "httpd restat..."
/usr/bin/ps aux I grep httpd
;;
status)
/usr/bin/systemctl $1 httpd
;;
*)
echo "plases input start | stop | restat | status"
esac