Shell 第二章《流控》
前言
无论什么编程语言都离不开条件判断(流控)。SHELL也不例外。
例如,用户输入的密码不够长时提示用户,你太短了
例如,用户输入了备份的目录,如果有目录继续备份,如果没有目录创建目录。
例如,用户输入成绩,如果100-80分评为优秀,如果60-79分评为合格,如果59-0分评为不合格。
Shell 条件测试
数值比较 [ 整数1 操作符 整数2 ]
操作符
[ 20 -gt 10 ] 大于
[ 1 -lt 10 ] 小于
[ 1 -eq 1 ] 等于
[ 1 -ne 10 ] 不等于
[ 20 -ge 10 ] 大于等于
[ 10 -le 10 ] 小于等于
示例
需求:
猜测用户输入的密码,是否满足长度需求
变量长度:echo ${#pass}
测试语法:
格式1: test 条件表达式
格式2: [ 条件表达式 ]
格式3: [[ 条件表达式 ]]
if语法结构
read -p "请输入您的密码:" ps
if [ ${#ps} -lt 7 ];then
echo "您的密码太短!"
else
echo "您的密码真长!"
fi
测试
[root@localhost ~]# bash pass1.sh
请输入您的密码:123
您的密码太短!
[root@localhost ~]# bash pass1.sh
请输入您的密码:1234
您的密码太短!
[root@localhost ~]# bash pass1.sh
请输入您的密码:123456
您的密码太短!
[root@localhost ~]# bash pass1.sh
请输入您的密码:1234567
您的密码真长!
文件测试 [ 操作符 文件或目录 ]
操作符(了解)
-f filename : 当filename 存在时返回真
-b filename : 当filename 存在并且是块文件时返回真(返回0)
-d pathname : 当pathname 存在并且是一个目录时返回真
-h filename : 当filename 存在并且是符号链接文件时返回真 (或 -L filename)
-c filename : 当filename 存在并且是字符文件时返回真
-e pathname : 当由pathname 指定的文件或目录存在时返回真
-g pathname : 当由pathname 指定的文件或目录存在并且设置了SGID 位时返回真
-k pathname : 当由pathname 指定的文件或目录存在并且设置了"粘滞"位时返回真
-p filename : 当filename 存在并且是命名管道时返回真
-r pathname : 当由pathname 指定的文件或目录存在并且可读时返回真
-s filename : 当filename 存在并且文件大小大于0 时返回真
-S filename : 当filename 存在并且是socket 时返回真
-t fd: 当fd 是与终端设备相关联的文件描述符时返回真
-w pathname : 当由pathname 指定的文件或目录存在并且可写时返回真
-x pathname : 当由pathname 指定的文件或目录存在并且可执行时返回真
-O pathname : 当由pathname 存在并且被当前进程的有效用户id 的用户拥有时返回真(字母O 大写)
-G pathname : 当由pathname 存在并且属于当前进程的有效用户id 的用户的用户组时返回真
需求:
请用户输入备份的路径,如果存在提示已存在可以备份,如果不存在提示目录不存在请创建。
脚本
#!/bin/bash
read -p "请输入您要备份的目录:" cata
if [ -d $cata ];then
echo "$cata已存在可以备份"
else
echo "${cata}不存在无法创建"
fi
[root@bogon ~]# sh catalogue.sh
请输入您要备份的目录:ccc
ccc不存在无法创建
[root@bogon ~]# mkdir /soft
[root@bogon ~]# sh catalogue.sh
请输入您要备份的目录:/soft
/soft已存在可以备份
字符串比较 [ “字符串” = “字符串” ]
=,等于
需求
邀请用户确认,yes升级,否则不升级
脚本
#!/bin/bash
read -p "您确定要升级吗:" tt
if [ ${tt} = "yes" ];then
echo "正在升级"
else
echo "您不升级"
fi
[root@bogon ~]# sh zifu.sh
您确定要升级吗:yes
正在升级
[root@bogon ~]# sh zifu.sh
您确定要升级吗:no
[root@bogon ~]# 您不升级
!=,不等于
注意叹号和等号间没有空格
脚本
#!/bin/bash
read -p "您确定要升级吗:" tt
if [ ${tt} != "yes" ];then
echo "正在升级"
else
echo "您不升级"
fi
[root@bogon ~]# sh zifu.sh
您确定要升级吗:yes
您不升级
[root@bogon ~]# sh zifu.sh
您确定要升级吗:no
正在升级
发现正好和上面相反
-z:判断字符长度是为0(了解)
-n: 判断字符长度不是为0
双引号的重要性,解决一元表达式的问题
双引号的重要性,解决一元表达式的问题
[root@localhost ~]# BBB=""
[root@localhost ~]# echo ${#BBB}
0
[root@localhost ~]# [ -z "$BBB" ] 字符长度是为0
[root@localhost ~]# echo $?
0
[root@localhost ~]# [ -n "$BBB" ] 字符长度不为0
[root@localhost ~]# echo $?
1
and和or
简介
当条件测试比较复杂时,需要多个条件同时成立。就需要混合条件测试了。
逻辑的(and)与(or):&& 逻辑的 AND 的意思, -a ,两个条件同时成立,为真。
|| 逻辑的 OR 的意思, -o ,两个条件一个成立,为真。
事例
猜测用户输入的密码是否满足如下条件
1.长度大于等于7位
2.包含字母大写
3.包含字母小写
4.包含符号“@!_”
脚本
read -p "请输入您的密码:" password
if [ ${#password} -gt 7 ] && [[ ${password} =~ [a-z] ]] && [[ ${password} =~ [A-Z] ]] && [[ ${password} =~ [@_!] ]];then
echo "您的密码真的复杂"
else
echo “您的密码太简单了”
fi
[root@bogon ~]# sh andor.sh
请输入您的密码:12345aZ!
您的密码真的复杂
[root@bogon ~]# sh andor.sh
请输入您的密码:123245
“您的密码太简单了”
多种表达方法(了解)
[root@localhost ~]# [ 1 -lt 2 -a 5 -gt 10 ]
[root@localhost ~]# [ 1 -lt 2 -o 5 -gt 10 ]
[root@localhost ~]# [[ 1 -lt 2 && 5 -gt 10 ]]
[root@localhost ~]# [[ 1 -lt 2 || 5 -gt 10 ]]
[root@localhost ~]# [ 1 -lt 2 ] && [ 11 -gt 10 ]
流程控制 if
单分支结构
语法
if [ command/test ];then
符合该条件执行的语句
fi
需求
编写脚本,由用户输入用户名,如果用户不存在,则创建该用户
了解一下$?变量
上个命令的退出状态,或函数的返回值
#!/bin/bash
read -p "请输入要创建的用户名称:" name
id $name &> /dev/null
if [ $? -ne 0 ];then
useradd $name
fi
[root@bogon ~]# sh create
请输入要创建的用户名称:cccc
[root@bogon ~]# id cccc
uid=1025(cccc) gid=1025(cccc) 组=1025(cccc)
双分支结构
语法
if 条件测试
then
命令序列
else
命令序列
fi
需求
编写脚本,由用户输入用户名,如果用户不存在,则创建该用户,并设置密码为123456;否则,提示用户已经存在
脚本
#!/bin/bash
read -p "请输入要创建的用户名称:" name
if id $name &> /dev/null; then
echo "${name}用户已经存在"
else
useradd ${name}
echo "123456" | passwd --stdin $name &> /dev/null
echo "$name用户创建成功密码是123456"
fi
[root@bogon ~]# sh user.sh
请输入要创建的用户名称:rrr
rrr用户创建成功密码是123456
[root@bogon ~]# sh user.sh
请输入要创建的用户名称:rrr
rrr用户已经存在
多分支结构
语法
多分支结构
if 条件测试1
then 命令序列
elif 条件测试2
then 命令序列
elif 条件测试3
then 命令序列...
else 命令序列
fi
需求
编写脚本,取出系统时间的小时,对数字进行判断
6–10 this is morning
11-13 this is noon
14-18 this is afternoon
其他 this is night
脚本
#!/bin/bash
hour=`date +%H`
if [ $hour -ge 6 -a $hour -le 10 ];then
echo "this is morning"
elif [ $hour -ge 11 -a $hour -le 13 ];then
echo "this is noon"
elif [ $hour -ge 14 -a $hour -le 18 ];then
echo "this is afternoon"
else
echo "this is night"
fi
[root@bogon ~]# sh date.sh
this is afternoon
[root@bogon ~]# date
2022年 08月 24日 星期三 16:59:30 CST
[root@bogon ~]# date -s 10:10:06
2022年 08月 24日 星期三 10:10:06 CST
[root@bogon ~]# sh date.sh
this is morning
嵌套结构(了解)
语法
嵌套结构
if 条件测试1 then 命令序列
if 条件测试1 then 命令序列
else 命令序列
fi
else 命令序列
fi
需求
如图所示:创建用户,如果不存在则创建用户提示用户输入密码,密码大于7位则创用户,密码小于七位提示密码不符合要求。如果用户已经存在,提示存在即可!
脚本
#!/bin/bash
read -p "请输入要创建用户名称:" name
id ${name} &> /dev/null
if [ $? -eq 0 ];then
echo "${name}已存在"
else
useradd ${name}
echo "$name创建完成:"
read -p "请输入用户密码" password
if [ ${password} -ge 7 ];then
echo ${password} | passwd --stdin $name
echo "$name用户密码是${password}"
else
echo "密码不符合要求"
fi
fi
[root@bogon ~]# sh name.sh
请输入要创建用户名称:kkk
kkk创建完成:
请输入用户密码123456
更改用户 kkk 的密码 。
passwd:所有的身份验证令牌已经成功更新。
kkk用户密码是123456
[root@bogon ~]# sh name.sh
请输入要创建用户名称:kkk
kkk已存在
调试脚本
调试脚本的其他方法:
# sh -n 02.sh 仅调试脚本中的语法错误。
# sh -vx 02.sh 以调试的方式执行,查询整个执行过程
注意
1、[ ]表示条件测试。注意这里的空格很重要。要注意在”[“后面和前面”]都必须要有空格
2、在shell中,then和fi是分开的语句。如果要在同一行里面输入,则需要用分号将他们隔开。
3、注意if判断中对于变量的处理,需要加引号,以免一些不必要的错误。没有加双引号会在一些含空格等的字符串变量判断的时候产生错误。比如[ -n “$var” ]如果var为空会出错
4、判断是不支持浮点值的
5、如果只单独使用>或者<号,系统会认为是输出或者输入重定向,虽然结果显示正确,但是其实是错误的,因此要对这些符号进行转意
6、默认,运行if语句中的命令,所产生的错误信息。仍然出现在脚本的输出结果中
7、使用-z或者-n来检查长度的时候,没有定义的变量也为0
8、空变量和没有初始化的变量可能会对shell脚本测试产生灾难性的影响,因此在不确定变量的内容的时候,在测试号前使用-n或者-z测试一下
9、$? 变量包含了之前执行命令的退出状态(最近完成的前台进程)(可以用于检测退出状态)
模式匹配:case
前言
shell编程中if和case都是用来做流控的。
下面先通过一个案例1,来了解case的特点。
案例1:简单的模式匹配
需求
邀请用户输入待删除用户名,询问用户,确定要继续删除吗 yes/no: ” y
if写法
#!/bin/bash
#name
#time
#1请输入删除的用户名:
read -p "please input a username : " user
#2请用户确认是否删除
read -p "are you sure?[y/n]: " action
if [ "$action" = "y" -o "$action" = "Y" ] ;then
userdel -r $user
echo "$user is deleted!"
else
echo "thank you"
fi
case 语法结构
case 变量 in
模式1)
命令序列1
;;
模式2)
命令序列2
;;
模式3)
命令序列3
;;
*)
无匹配后命令序列
case写法
#!/bin/bash
#name
#time
#1请输入删除的用户名:
read -p "please input a username : " user
#2请用户确认是否删除
read -p "确认删除吗?[yes/no] " action
#3 case 流控写法
case "$action" in
Y|y|YES|yes|Yes|YeS|YEs)
userdel -r $user
echo "$user is deleted!"
;;
*)
echo "thank you"
;;
esac
案例2:简单的JumpServer
需求
由于工作中,我们需要管理N多台服务器。那么访问服务器就是一件繁琐的事情。通过shell编程,编写跳板程序,当我们需要访问服务器时,看一眼服务器列表名,按一下数字,就登录成功了。
跳板主机,运行跳板脚本。弹出数十台服务器名的菜单
1)mysql1
2)mysql2
3)bj-web1
........
h) help
q) exit
用户请选择要连接的主机[1-3]: 1
Last login: Sun Sep 6 04:18:01 2015 from 192.168.122.1
[root@localhost ~]$连接成功!!!
演示
#!/usr/bin/bash
#定义目标主机IP
web1=192.168.51.152
web2=192.168.51.128
mysql1=192.168.51.88
#打印跳转菜单
cat <<EOF
1.WEB1
2.WEB2
3.MYSQL1
EOF
#读取用户输入
read -p "input number: " num
#判断用户选择
case $num in
1)
ssh alice@$web1
;;
2)
ssh alice@$web2
;;
*)
echo "123"
esac
案例3:系统管理工具箱
前言
linux提供的丰富的管理命令,用户管理,内存管理,磁盘管理,进程管理,日志管理,文件管理,软件管理,网络管理等等数十个工具包。
如果你能通过shell编程,把他们编写到一个程序里。想用某些功能,只需要按回车,就能完成。
事例
Command action
h 显示命令帮助
f 显示磁盘分区
d 显示磁盘挂载
m 查看内存使用
u 查看系统负载
q 退出程序
Command (h for help): m
total used free shared buffers cached
Mem: 7628 840 6788 0 29 378
Swap: 2047 0 2047
演示
定义脚本
1 输出菜单并测试。
vim systemmanage.sh
#!/usr/bin/bash
#打印菜单
cat <<-EOF
h. help
f. disk partation
d. filesystem mount
m. memory
u. system load
q.exit
EOF
# 读取用户输入,进行模式匹配
read -p "please input [h for help]: " action
case "$action" in
f)
fdisk -l
;;
d)
df -hT
;;
m)
free -m
;;
u)
uptime
;;
q)
exit
;;
"")
;;
*)
echo "error"
;;
esac