(六)shell编程之循环控制语句

循环

将某代码段重复运行多次,通常有进入循环的条件和退出循环的条件
重复运行次数

循环次数事先已知

循环次数事先未知

常见的循环的命令:for, while, until

循环 for

格式1:

for NAME [in WORDS ... ] ; do COMMANDS; done

#方式1
for 变量名 in 列表;do
   循环体
done

#方式2
for 变量名 in 列表
do
   循环体
done

执行机制:
依次将列表中的元素赋值给“变量名”; 每次赋值后即执行一次循环体; 直到列表中的元素耗尽,循环结束

如果省略 [in WORDS ... ] ,此时使用位置参数变量 in "$@"
for 循环列表生成方式:
直接给出列表
整数列表:

{start..end}
$(seq [start [step]] end)

返回列表的命令:

$(COMMAND)

使用glob,如:*.sh

变量引用,如:$@,$*,$#

范例:面试题,计算1+2+3+...+100 的结果

[root@centos8 ~]# sum=0;for i in {1..100};do let sum+=i;done ;echo sum=$sum
sum=5050
[root@centos8 ~]# seq -s+ 100|bc5050
5050
[root@centos8 ~]# echo {1..100}|tr ' ' +|bc
5050
[root@centos8 ~]# seq 100|paste -sd +|bc
5050

范例: 100以内的奇数之和

[root@centos8 ~]# sum=0;for i in {1..100..2};do let sum+=i;done;echo sum=$sum
sum=2500
[root@centos8 ~]# seq -s+ 1 2 100| bc
2500
[root@centos8 ~]# echo {1..100..2}|tr ' ' + | bc
2500

范例: 批量创建用户

#!/bin/bash
# 
#********************************************************************
#Author:        xuanlv
#QQ:            360956175
#Date:          2021-06-10
#FileName:     create_user.sh
#URL:           https://www.cnblogs.com/xuanlv-0413/
#Description:      The test script
#Copyright (C):     2021 All rights reserved
#********************************************************************
[ $# -eq 0 ] && { echo "Usage: createuser.sh USERNAME ..."; exit 1; }
for user ;do
   id $user &> /dev/null && echo $user is exist || { useradd $user ; echo $user is created; }
done

范例: 批量创建用户和并设置随机密码

#!/bin/bash
# 
#********************************************************************
#Author:        xuanlv
#QQ:            360956175
#Date:          2021-06-10
#FileName:     user_for.sh
#URL:           https://www.cnblogs.com/xuanlv-0413/
#Description:      The test script
#Copyright (C):     2021 All rights reserved
#********************************************************************
for i in {1..10};do
  useradd user$i
  PASS=`cat /dev/urandom | tr -dc '[:alnum:]' | head -c12`
  echo $PASS | passwd --stdin user$i &> /dev/null
  echo user$i:$PASS >> /data/user.pass
  echo "user$i is created"
done

范例:面试题,要求将目录YYYY-MM-DD/中所有文件,移动到YYYY-MM/DD/下

#1.创建YYYY-MM-DD格式的目录,当前日期一年前365天到目前共365个目录,里面有10个文件.log后缀的
文件
#!/bin/bash
# Date: 2021-06-10
# Author: lvxuan
PDIR=/data/tests
for i in {1..365};do
  DIR=`date -d "-$i day" +%F`
  mkdir -p $PDIR/$DIR
  cd $PDIR/$DIR
  for j in {1..10};do
     touch $RANDOM.log
  done
done

#2.将上面的目录移动到YYYY-MM/DD/下
#!/bin/bash
# Date: 2021-06-10
# Author: lvxuan
DIR=/data/tests
cd $DIR || { echo 无法进入 $DIR;exit 1; }
for subdir in * ;do
  YYYY_MM=`echo $subdir | cut -d"-" -f1,2`
  DD=`echo $subdir | cut -d"-" -f3`
  [ -d $YYYY_MM/$DD ] || mkdir -p $YYYY_MM/$DD &> /dev/null
  mv $subdir/* $YYYY_MM/$DD
done
rm -rf $DIR/*-*-*

#执行
[root@localhost ~]# bash for_dir.sh
[root@localhost ~]# ll /data/tests/
[root@localhost ~]# bash sum_dir.sh
[root@localhost ~]# tree /data/tests/

格式2

双小括号方法,即((…))格式,也可以用于算术运算,双小括号方法也可以使bash Shell实现C语言风格
的变量操作
I=10;((I++))

for ((: for (( exp1; exp2; exp3 )); do COMMANDS; done

for ((控制变量初始化;条件判断表达式;控制变量的修正表达式))
do
  循环体
done

说明:
控制变量初始化:仅在运行到循环代码段时执行一次
控制变量的修正表达式:每轮循环结束会先进行控制变量修正运算,而后再做条件判断

循环 while

格式:

while COMMANDS; do COMMANDS; done

while CONDITION; do
  循环体
done

说明:
CONDITION:循环控制条件;进入循环之前,先做一次判断;每一次循环之后会再次做判断;条件为
“true”,则执行一次循环;直到条件测试状态为“false”终止循环,因此:CONDTION一般应该有循环控
制变量;而此变量的值会在循环体不断地被修正

进入条件:CONDITION为 true

退出条件:CONDITION为 false

无限循环

while true; do
  循环体
done

while : ; do
  循环体
done

范例:

[root@centos8 ~]# sum=0;i=1;while ((i<=100));do let sum+=i;let i++;done;echo $sum
5050

范例:监控磁盘使用率

[root@centos8 ~]# cat while_diskcheck.sh
#!/bin/bash
# Date: 2021-06-10
# Author: lvxuan
WARNING=80
while :;do
  USE=`df | sed -rn '/^/dev/sd/s#.* ([0-9]+)%.*#1#p' | sort -nr | head -n1`
  if [ $USE -gt $WARNING ];then
     echo Disk will be full from `hostname -I` | mail -s "disk warning" xxx@qq.com
  fi
  sleep 10
done

范例: 防止Dos攻击的脚本

#!/bin/bash
# Date: 2021-06-10
# Author: lvxuan
WARNING=10
touch deny_hosts.txt
while true;do
   ss -nt | sed -nr '1!s#.* ([0-9.]+):[0-9]+ *#1#p' | sort | uniq -c | sort |
while read count ip;do
      if [ $count -gt $WARNING ];then
        echo $ip is deny
        grep -q "$ip" deny_hosts.txt || { echo $ip >> deny_hosts.txt; iptables -A INPUT -s $ip -j REJECT; }
      fi
   done
   sleep 10
done

循环 until

格式:

until COMMANDS; do COMMANDS; done

until CONDITION; do
  循环体
done

说明:
进入条件: CONDITION 为false
退出条件: CONDITION 为true

范例:

[root@centos8 ~]# sum=0;i=1;until ((i>100));do let sum+=i;let i++;done;echo $sum
5050

无限循环

until false; do
   循环体
Done

循环控制语句 continue

continue [N]:提前结束第N层的本轮循环,而直接进入下一轮判断;最内层为第1层

格式:

while CONDITION1; do
  CMD1
  ...
  if CONDITION2; then
     continue
  fi
  CMDn
  ...
done

范例:

#!/bin/bash
# Date: 2021-06-10
# Author: lvxuan
for ((i=0;i<10;i++));do
   for((j=0;j<10;j++));do
      [ $j -eq 5 ] && continue 2
      echo $j
   done
   echo ---------------------------
done

输出结果:
0
1
2
3
4
0
1
2
3
4
0
1
2
3
4

循环控制语句 break

break [N]:提前结束第N层整个循环,最内层为第1层

格式:

while CONDITION1; do
  CMD1
  ...
  if CONDITION2; then
     break
  fi
  CMDn
  ...
done

范例:

#!/bin/bash
# Date: 2021-06-10
# Author: lvxuan
for ((i=0;i<10;i++));do
   for((j=0;j<10;j++));do
       [ $j -eq 5 ] && break 2
       echo $j
   done
   echo ---------------------------
done

# 执行
[root@centos8 script]# bash break_for.sh
0
1
2
3
4

范例:

#!/bin/bash
# Date: 2021-06-10
# Author: lvxuan
NUM=$[RANDOM%10]
while read -p "输入 0-9 之间的数字: " INPUT ;do
  if [ $INPUT -eq $NUM ];then
     echo "恭喜你猜对了!"
     break
  elif [ $INPUT -gt $NUM ];then
     echo "数字太大了,重新猜!"
  else
     echo "数字太小了,重新猜!"
  fi
done

循环控制 shift 命令

shift [n] 用于将参量列表 list 左移指定次数,缺省为左移一次。
参量列表 list 一旦被移动,最左端的那个参数就从列表中删除。while 循环遍历位置参量列表时,常用到 shift

范例:

#!/bin/bash
# Date: 2021-06-10
# Author: lvxuan
while [ $You can't use 'macro parameter character #' in math mode# -gt 0 ] 
do
  echo $*
  shift
done

./doit.sh a b c d e f g h

范例:

#!/bin/bash
# Date: 2021-06-10
# Author: lvxuan
until [ -z "$1" ]
do
  echo "$1"
  shift
done
echo

./shfit.sh a b c d e f g h

while 特殊用法 while read

while 循环的特殊用法,遍历文件或文本的每一行

格式:

while read line; do
   循环体
done < /PATH/FROM/SOMEFILE

说明:依次读取/PATH/FROM/SOMEFILE文件中的每一行,且将行赋值给变量line

范例:

[root@centos8 ~]# echo long | read X ; echo $X
[root@centos8 ~]# echo long | while read X ; do echo $X;done
long
[root@centos8 ~]# echo long | { read X ; echo $X; }
long
[root@centos8 ~]# echo long | ( read X ; echo $X )
long
[root@centos8 ~]# echo long wang zhang | ( read X Y Z; echo $X $Y $Z )
long wang zhang
[root@centos8 ~]# echo long wang zhang | while read X Y Z; do echo $X $Y $Z;done
long wang zhang

范例:

#!/bin/bash
# Date: 2021-06-10
# Author: lvxuan
WARNING=80
MAIL=root@xxx.com
df |sed -nr "/^/dev/sd/s#^([^ ]+) .* ([0-9]+)%.*#1 2#p"| while read DEVICE
USE;do
  if [ $USE -gt $WARNING ] ;then
     echo "$DEVICE will be full,use:$USE" | mail -s "DISK WARNING" $MAIL
  fi
done

范例:

#!/bin/bash
# Date: 2021-06-10
# Author: lvxuan
MAX=3
lastb | sed -rn '/ssh:/s@.* ([0-9.]{1,3}{3}[0-9]{1,3}) .*@1@p'| sort | uniq -c
| while read count ip ;do
  if [ $count -gt $MAX ];then
     iptables -A INPUT -s $ip -j REJECT
  fi
done

范例:查看/sbin/nologin的shell类型的用户名和UID

#!/bin/bash
# Date: 2021-06-10
# Author: lvxuan
while read line ;do
  if [[ "$line" =~ /sbin/nologin$ ]] ;then
     echo $line | cut -d: -f1,3
  fi
done < /etc/passwd

循环与菜单 select

格式:

select NAME [in WORDS ... ;] do COMMANDS; done

select variable in list ;do
  循环体命令
done

说明:

select 循环主要用于创建菜单,按数字顺序排列的菜单项显示在标准错误上,并显示 PS3 提示
符,等待用户输入
用户输入菜单列表中的某个数字,执行相应的命令
用户输入被保存在内置变量 REPLY 中
select 是个无限循环,因此要用 break 命令退出循环,或用 exit 命令终止脚本。也可以按 ctrl+c
退出循环
select 经常和 case 联合使用
与 for 循环类似,可以省略 in list,此时使用位置参量
原文地址:https://www.cnblogs.com/xuanlv-0413/p/14881436.html