Shell编程之循环语句
目录
for循环语句
for语句的结构
for语句应用示例
根据姓名列表批量添加用户
根据IP地址列表检查主机状态
使用while循环语句
while语句的结构
while语句应用示例
批量添加规则编号的用户
猜价格游戏
until循环语句
until语句的结构
until语句应用示例
计算1-50的和
为指定用户发送在线消息
for循环语句
在实际工作中,经常会遇到某项任务需要多次执行的情况,而每次执行时仅仅是处理的对象不一样,
其他命令相同当面对各种列表重复任务时,使用简单的if语句已经难以满足要求,而顺序编写全部代码更是显得异常烦琐、困难重重
for语句的结构
使用for循环语句时,需要指定一个变量及可能的取值列表,针对每个不同的取值重复执行相同的
命令序列,直到变量值用完退出循环。在这里,"取值列表"称为for语句的执行条件,其中包括多个
属性相同的对象,需要预先指定(如通讯录、IP黑名单)。
for变量名in取值列表
do
命令序列
done
上述语句结构中,for语句的操作对象为用户指定名称的变量,并通过 in关键字为该变量预先设置
了一个取值列表,多个取值之间以空格进行分隔。位于do...done之间的命令序列称为循环体,其中的执行语句需要引用变量以完成相应的任务
for语句的执行流程:首先将列表中的第一个取值赋给变量,并执行do...done循环体中的命令序列;
然后将列表中的第二个取值赋给变量,并执行循环体中的命令冷序列.....依此类推,直到列表中的所有取值用完,最后将跳至done语句,表示结束循环
for语句应用实例
根据姓名列批量添加用户
根据人事部门给出的员工姓名的拼音列表,在Linux服务器中添加相应的用户账号,初始密码均设
置为"123456"。其中,员工姓名列表中的账号数量并不固定,而且除了要求账号名称是拼音之外,并无其他特殊规律。针对上述要求,可先指定员工列表文件users.txt,然后编写一个名为uaddfor.sh 的Shell脚本,从users.txt文件中读取各用户名称,重复执行添加用)户、设置初始密码的相关操作
[root@localhost ~]# vim /root/users.txt
chenye
dengchao
zhangjie
[root@localhost ~]# vim uaddfor.sh//批量添加用户的脚本
#!/bin/bash
ULIST=$(cat /root/users.txt)
for UNAME in $ULIST
do
useradd $UNAME
echo "123456"
passwd --stdin $UNAME &>/dev/null
done
[root@localhost ~]# chmod +x uaddfor.sh
[root@localhost ~]# ./uaddfor.sh
[root@localhost ~]# tail -3 /etc/passwd
chenye:x:1005:1005:1005: /home/chenye:/bin/bash
dengchao:x:1006:1006:1006: /home/dengchao:/bin/bash
zhangjie:x:1007:1007:1007::/home/zhangjie:/bin/bash
若要删除uaddfor.sh脚本所添的用户,只需将for循环体中添加的用户的命令改为序列删除用户的操作即可,比如建立一个udelfor.sh脚本
[root@localhost ~]# vim udelfor.sh
#!/bin/bash
ULIST=$(cat /root/users.txt)
for UNAME in $ULIST
do
userdel -r $UNAME &>/dev/null
done
[root@localhost ~]# chmod +x udelfor.sh
[root@localhost~]#./udelfor.sh
[root@localhost ~]# id chenye
根据IP地址列表检查主机状态
根据包含公司各服务器IP地址的列表文件,检查其中各主机的ping连通性,输出各主机是否启动、
关闭。其中,服务器的数量并不固定,各服务器的IP地址之间也无特殊规律
针对此案例要求,可先指定IP地址列表文件ipadds.txt,勿后编写一个名为chkhosts.sh的
Shell脚本,从ipadds.txt文件中读取各服务器的IP地址,重复执行ping连通性测试,并根据测
试结果输出相应的提示信息
[root@localhost ~]# vim /root/ipadds.txt
172.16.16.1
172.16.16.22
172.16.16.220
[root@localhost~]#vim chkhosts.sh
#!/bin/bash
HLIST=$(cat /root/ipadds.txt)
for IP in $HLIST
do
ping -C 3 -i 0.2 -W 3 $IP &> /dev/null
if [ $? -eq 0 ]
then
echo "Host $IP is up."
else
echo "Host $IP is down."
fi
done
[root@localhost ~]# chmod +x chkhosts.sh
[root@localhost ~]#./chkhosts.shHost 172.16.16.1 is up.
Host 172.16.16.22 is up.
Host 172.16.16.220 is down.
使用while循环语句
for循环语句非常适用于列表对象无规律,且列表来源已固定(如某个列表文件)的场合。而对于要
求控制循环次数、操作对象按数字顺序编号、按特定条件执行重复操作等情况,则更适合使用另外一种循环-while语句
while语句的结构
使用while循环语句时,可以根据特定的条件反复执行一个命令序列,直到该条件不再满足时为止。
在脚本应用中,应该避免出现死循环的情况,否则后边的命令操作非将无法执行。因此,循环体内的命令序列中应包括修改测试条件的语句,以便在适当的时候使测试条件不不再成立,从而结束循环
结构图
while 条件测试
do
命令序列
done
while语句的执行流程:首先判断while后的条件测试操作结果,如果条件成立,则执行do...done
循环体中的命令序列;返回while后再次判断条件测试结果,如果条件仍然成立,则继续执行循环体;
再次返回到while后,判断条件测试结果.......如此循环,直到while后的条件测试结果不再成立为止,最
后跳转到done语句,表示结束循环使用while循环语句时,有两个特殊的条件测试操作,即true(真)和false(假)。使用 true
作为条件时,表示条件永远成立,循环体内的命令序列将无限执行下去,除非强制终止脚本(或通过exit
语句退出脚本);反之,若使用false作为条件,则循环体将不会被执行。这两个特殊条件也可以用在if
语句的条件测试中
while语句应用示例
批量添加规律编号的用户
在一些技术培训和学习领域,出于实验或测试的目的,需要要批量添加用户账号,这些用户的名称中包
含固定的前缀字串,并按照数字顺序依次进行编号,账号的为数量往往也是固定的
[root@localhost ~]# vim uaddwhile.sh
#!/bin/bash
PREFIX="stu"
i=1
while [ $i -le 20 ]
do
useradd ${PREFIX}$i
echo "123456" | passwd --stdin ${PREFIX}$i &> /dev/null
let i++done
[root@localhost ~]# chmod +x uaddwhile.sh
上述脚本代码中,使用变量i来控制用户名称的编号,初始赋值为1,并且当取值大于20时终止
循环。在循环体内部,通过语句"let i++"(等同于i=2xpr$i+1`)来使变量i的值增加1,因
此当执行第一次循环后i的值将变为2,执行第二次循环后i的值将变为3,........依此类推。
测试并确认 uaddwhile.sh脚本的执行结果
[root@localhost ~]# ./uaddwhile.sh
[root@localhost ~]# grep "stu" /etc/passwd | tail -3
stu18:x:1022:1022::/home/stu18:/bin/bash
stu19:x:1023:1023::/home/stu19:/bin/bash
stu20:x:1024:1024::/home/stu20:/bin/bash
若要删除所添用户,将while循环体中添加用户改为删除用户即可
[root@localhost ~]# vim udelwhile.sh
#!/bin/bash
PREFIX="stu"
i=1
while [ $i -le 20 ]
do
userdel -r ${PREFIX}$i
done
[root@localhost ~]# chmod +x udelwhile.sh
[root@localhost~]#./udelwhile.sh
[root@localhost ~]# id stu20
猜价格游戏
随机生成一个脚本,用户判断价格是否高或者低,给出相应的提示一直到猜出来为止,输入用户共猜的次数,实际价格通过环境变量RANDOM可以获得小于216的随机整数计算与其中1000的余数可以获得0-999的随机价格,反复测试可以通过true作为测试条件的while循环实现,
[root@localhost ~]# vim pricegame.sh
#!/bin/bash
PRICE=$(expr $RANDOM % 1000)
TIMES=0
echo"商品实际价格范围为0-999,猜猜看是多少?"
while true
do
read-p"请输入你猜测的价格数目:"INT
let TIMES++
if [ $INT -eq $PRICE ]
then
echo"恭喜你答对了,实际价格是$PRICE"
echo"你总共猜测了$TIMES次"
exit 0
elif [ $INT -gt $PRICE ]
then
echo"太高了!"
elseecho"太低了“
if
done
[root@localhost ~]# chmod +x pricegame.sh
[root@localhost ~]# ./pricegame.sh
商品实际价格范围为0-999,猜猜看是多少?
请输入你猜测的价格数目:500
太高了!
请输入你猜测的价格数目:250
太低了!
请输入你猜测的价格数目:375
太高了!
请输入你猜测的价格数目:280
太高了!
请输入你猜测的价格数目:265
太高了!
请输入你猜测的价格数目:253
恭喜你答对了,实际价格是253
你总共猜测了6次
until循环语句
until语句的结构
until循环与while循环类似,while循环能实现的脚本until同样也可以实现,但区别是while
循环在条件为真是继续执行循环,而until则是在条件为假时执行循环
语法结构
until条件测试操作
do
命令序列
done
until语句的执行流程:首先判断until后的条件测试操作结果如果条件不成立,则执行do...done
循环体中的命令序列:返回until后再次判断条件测试结结果,如果条件仍然不成立,则继续执行循环体;
再次返回到 until后,判断条件测试结果.......如此循环,直到 until后的条件测试结果成立为止,最后跳
转到done语句,表示结束循环
until语句应用示例
计算1-50的和
在一些科学计算领域,经常会用到各种数的计算,自然数的求和操作是最简单的例子
[root@localhost ~]# vim sumito50_until_v1.sh
#!/bin/bash
i=0;s=0
until [ $i -eq 50 ]
dolet "i=$i+1";let "s=$s+$i" done
echo 'sum(1..50)='$s
结果
[root@localhost ~]# chmod +x sumito50_until_v1.sh
[root@localhost ~]# ./sum1to50_until_v1.sh
sum(1..50)=1275
为指定用户发送在线消息
公司内部有一台Linux测试服务器,开发、测试、运维都在使用自自己的账号连接登录到服务器上。
当业务增加不能满足使用需求时,运维决定给服务器增加内存配置,要通知开发和测试人员保存数据退出,之后再关机升级内存,以应对业务的增加可以编写一个until-user_online to write.sh 的 Shell脚本,用于给已登录用户发送消息,对用户进行检查,必须是系统内用户并且处于登录状态
[root@localhost ~]# vim until-user_online_to_write.sh
#!/bin/bash
if [ $# -lt 2 ]; then
echo "Usage: $0 <username1> <username2> ... <message>"
exit 1
fi
message="${!#}"
for username in "${@:1:$#-1}"; do
if!grep -q "^$username:"/etc/passwd;then
echo "$username is not a valid user on this system."
continue
fi
while ! who | grep -q "$username"; do
echo "$username is not logged on. Waiting for the user to log in...
sleep 60
done
echo "Sending message to $username..."
write "$username" <<EOF
$message
EOF
if [ $? -ne 0 ]; then
echo "Failed to send message to $username."
fi
done
[root@localhost ~]#chmod +x until-user_online_towrite.sh
[root@localhost ~]# ./until-user_online_to_write.sh root
//发给 root自己,消息为空
Message from root@localhost on pts/0 at 16:23... Are you ready ? root
EOF
[root@localhost ~]# ./until-user_online_to_write.sh root hello
//发给root自己,消息为"hello"
Message from root@localhost on pts/0 at 16:23 ...hello
EOF
[root@localhost ~]# ./until-user_online_to_write.sh jerry hello
//发给用户 jerry,消息为"hello"
[root@localhost_~]# ./until-user_online_to_write.sh jerry
//发给用户 jerry,消息为空