linux shell循环:for、while、until用法详解

学习脚本就是为了方便我们运维工作,而对运维人员从来讲,有很多工作也是需要重复执行的,比如我们管理用户帐户时就需要批量创建和删除用户帐户。而此时如果使用系统自带的useradd或者userdel去操作的话,就只能机械式的一个一个创建,一个一个的删除。那我们是不是就可以考虑写一个程序,让它自动化的去创建了。此时,我们的循环执行概念也就出来了。循环就是让程序按照一定的条件反复的执行相关操作,直到不在满足循环条件时结束。

循环语句常见分类:

1,for语句
2,while语句
3,until语句

循环体包括两部分:
1,循环的进入条件
2,循环的退出条件

循环次数:
1,事先已知 for语句
2,事先未知 while、until语句

for循环常见语法格式:

for 变量名 in 列表;do
循环体
done

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

列表生成方式:
(1) 直接给出列表
(2) 整数列表:
(a) {start..end} 如:{1..10}
(b) $(seq [start [step]] end) 如:`seq 1 2 100` 表示1-100之间的所有奇数
(3) 返回列表的命令 $(COMMAND) 如:`ls /etc`
(4) 使用glob,如:*.sh
(5) 变量引用;$@, $*

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

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

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

1,计算1-100之间所有正整数的和

#!/bin/bash
#Author:wangjun
#Contact QQ:183530300
#Version:1.0
#Create time:2016-08-17 16:53:28
#Description:1-100 all positive integer's sum

declare -i sum=0

for I in {1..100};do
 let sum+=$I
done
echo "1-100 all positive integer's sum : $sum"

2,自动扫描1-254网段的所有主机,统计在线主机数与不在线主机数

#!/bin/bash 

#Author:wangjun
#Contact QQ:183530300
#Version:1.0
#Create time:2016-08-17 11:25:37
#Description:ping all hosts and count the number

declare -i online=0
declare -i offline=0
declare -i host=1 

for host in `seq 254`;do
 ping -c 1 -W 1 10.1.250.$host &> /dev/null && echo "10.1.250.$host is online" && let online=$online+1 || let offline++
done
echo "Online hosts number : $online"
echo "Offline hosts number : $offline"

3,生成10个随机数,并找出最大随机数和最小随机数

#!/bin/bash

#Author:wangjun
#Contact QQ:183530300
#Version:1.0
#Create time:2016-08-17 17:27:45
#Description:random number comparative

minrandom=$RANDOM
maxrandom=$minrandom

echo $minrandom

for I in {1..9};do
 RD=$RANDOM
 echo $RD
 if [[ $RD -gt $maxrandom ]];then
  maxrandom=$RD
 elif [[ $RD -lt $minrandom ]];then
  minrandom=$RD
 fi
done
echo "Minrandom is : $minrandom"
echo "Maxrandom is : $maxrandom"

4,打印九九乘法表(第一版)

#!/bin/bash
#Author:wangjun
#Contact QQ:18353030
#Version:1.0
#Create time:2016-08-16 13:07:40
#Description:99 multiplication table
for Row in {1..9};do
 for Column in `seq $Row`;do
  echo -ne "${Column}x${Row}=$[$Row*$Column]t"
 done
 echo
done

5,打印九九乘法表(第二版)——for循环特殊用法

#!/bin/bash

#Author:wangjun
#Contact QQ:183530300
#Version:1.0
#Create time:2016-08-18 11:35:16
#Description:multiplication table 

for ((row=1;row<=9;row++));do
 for ((column=1;column<=$row;column++));do
  echo -ne "${column}x${row}=$[${column}*${row}]t"
 done
 echo
done

循环执行 :将某代码段重复运行多次; 重复运行多少次: 循环次数事先已知 ;循环次数事先未知 ;有进入条件和退出条件。

三种循环体:for、while、until。

再循环前前介绍步进,在循环中经常用到步进。

步进

显示1-10

[root@centos6mini app]# echo {1..10}
1 2 3 4 5 6 7 8 9 10
[root@centos6mini app]# echo $(seq 1 10)
1 2 3 4 5 6 7 8 9 10

1-10 递加

[root@centos6mini app]# echo {1..10..2}
1 3 5 7 9
[root@centos6mini app]# echo $(seq 1 2 10)
1 3 5 7 9

1-10 递减

[root@centos6mini app]# echo {10..1..2}
10 8 6 4 2
[root@centos6mini app]# echo $(seq 10 -2 1)
10 8 6 4 2

1-100间的奇数和偶数

[root@centos6mini app]# echo {1..100..2}
1 3 5 7 9 11 13 15 17 19 21 23 25 27 29 31 33 35 37 39 41 43 45 47 49 51 53 55 57 59 61 63 65 67 69 71 73 75 77 79 81 83 85 87 89 91 93 95 97 99
[root@centos6mini app]# echo {2..100..2}
2 4 6 8 10 12 14 16 18 20 22 24 26 28 30 32 34 36 38 40 42 44 46 48 50 52 54 56 58 60 62 64 66 68 70 72 74 76 78 80 82 84 86 88 90 92 94 96 98 100
[root@centos6mini app]# echo {100..1..2}
100 98 96 94 92 90 88 86 84 82 80 78 76 74 72 70 68 66 64 62 60 58 56 54 52 50 48 46 44 42 40 38 36 34 32 30 28 26 24 22 20 18 16 14 12 10 8 6 4 2
[root@centos6mini app]# echo {99..1..2}
99 97 95 93 91 89 87 85 83 81 79 77 75 73 71 69 67 65 63 61 59 57 55 53 51 49 47 45 43 41 39 37 35 33 31 29 27 25 23 21 19 17 15 13 11 9 7 5 3 1

随机生成16位包含数字、大小写字母、符号的密码

cat /dev/urandom |tr -dc '0-9a-zA-Z!@_#?.,' |head -c 16

tr -d 和 tr -dc 的用法

[root@centos6mini app]# echo 12324345 |tr -d 2 #不要数字2
134345
[root@centos6mini app]# echo 12324345 |tr -dc 2 #支取数字2
22

for循环

for 变量名 in 列表;do
循环体
done

执行机制:

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

for循环的列表生成方式:

(1) 直接给出列表

(2) 整数列表:

 (a) {start..end}

 (b) $(seq [start [step]] end)

(3) 返回列表的命令 $(COMMAND)

(4) 使用glob,如:*.sh

(5) 变量引用; $@, $*

批量创建user1至user10,要求设置随机16位密码,包含数字、大小写字母、符号。并要求用户使用密码首次登录后,强制设置新密码。

#!/bin/bash
> /root/user.log
for i in {1..10};do
#创建十个用户
    useradd user$i && echo user$i is created
#随机生成密码
    password=$(cat /dev/urandom |tr -dc '0-9a-zA-Z!@_#?.,' |head -c 16)
#将用户和密码导出
    echo user$i:$password >> /root/user.log
#设置密码
    echo $password |passwd --stdin user$i &> /dev/null
#强制下次登录修改密码
    passwd -e user$i &> /dev/null
done

使用♥,利用for,显示出一个等腰三角形。

方法一:

for ((i=1;i<10;i++))
do
    for ((j=1;j<=i;j++))
do
        echo -n "♥"
    done
    echo
done
for ((i=1;i<9;i++))
do
    for ((j=9;j>i;j--))
do
        echo -n "♥"
    done
    echo
done

♥
♥♥
♥♥♥
♥♥♥♥
♥♥♥♥♥
♥♥♥♥♥♥
♥♥♥♥♥♥♥
♥♥♥♥♥♥♥♥
♥♥♥♥♥♥♥♥♥
♥♥♥♥♥♥♥♥
♥♥♥♥♥♥♥
♥♥♥♥♥♥
♥♥♥♥♥
♥♥♥♥
♥♥♥
♥♥
♥

方法二:

#!/bin/bash2
read -p "请输入三角形的高度: " num
for i in `seq $num`;do
    for x in `seq 1 $[$num-$i]`;do
        echo -e " c"
    done
    for n in `seq 1 $[2*$i-1]`;do
        #yanse=`tr -dc '1-6' < /dev/urandom | head -c 1`
        #echo -e "e[3"$yanse"m♥e[0mc"
        echo -e "e[34m♥e[0mc"
    done
    echo 
done
  ♥    
 ♥♥♥
 ♥♥♥♥♥
♥♥♥♥♥♥♥

方法三:

#!/bin/bash
for ((i = 1; i < 10; i++))
do
  for ((j = 10; j > i; j--))
  do 
   echo -n " ";
  done
  for ((m = 1; m <= i; m++))
  do 
    echo -n "$i "
  done
  echo ""
done
[root@centos7 ~]# ./sjx 
     1 
    2 2 
    3 3 3 
   4 4 4 4 
   5 5 5 5 5 
  6 6 6 6 6 6 
  7 7 7 7 7 7 7 
 8 8 8 8 8 8 8 8 
 9 9 9 9 9 9 9 9 9

方法四:

#!/bin/bash
#彩色闪烁山角形
read -p "请输入三角形边长: " w
for (( i=1;i<=$w;i++ )); do
    for(( j=$w;j>$i;j-- ));do
        echo -n " "
    done
    for (( m=1;m<=i;m++  ));do
        we=`cat /dev/urandom |tr -dc '1-6' |head -c 1 `
        echo -e "33[3"$we";5m♥33[0mc"  #红色加闪烁
        done
    echo

done

添加10个用户user1-user10,密码为8位随机字符、

#!/bin/bash
> /root/user.log
for i in {1..10};do  
  useradd user$i && echo user$i is created
  pass=$(cat /dev/urandom |tr -dc '0-9a-zA-Z!@_#?.,' |head -c 16)  #生成随机数
  echo user$i:---pass:$pass >> /root/user.log
  echo $pass |password --stdin user$i &> /dev/null
done

/etc/rc.d/rc3.d目录下分别有多个以K开头和以S开头的文件;分别读取每个文件,以K开头的输出为文件加stop,以S开头的输出为文件名加start,如K34filename stop S66filename start

#!/bin/bash
for i in $(ls /etc/rc.d/rc3.d/);do  #查看文件内容确定循环次数
    if [[ $i =~ ^K.* ]];then
        echo "$i stop"
    elif [[ $i =~ ^S.* ]];then
        echo "$i start"
    fi
done
[root@centos6mini app]# ./linshi 
K05wdaemon stop
K10saslauthd stop
K15htcacheclean stop
K15httpd stop
K30spice-vdagentd stop
K50dnsmasq stop
K50kdump stop
K73winbind stop
K74ntpd stop
...省略...

编写脚本,提示输入正整数n的值,计算1+2+…+n的总和

#!/bin/bash
read -p "请输入一个正整数" s
w=0
if [[ "$s" =~ ^[0-9]+$ ]] ;then
  for i in $(seq 1 $s) ;do
  w=$[$w+$i]           
  
  done
  echo "整数1到$s的和为$w"
else echo "请输入正整数"
fi

计算100以内所有能被3整除的整数之和

方法一:

[root@centos6mini app]# echo {3..100..3} |tr " " "+"|bc #进行步进,替换相加
1683

方法二:

#!/bin/bash
for i in {3..100..3};do  #步进,相加
 n=$[$n+$i] &> /devnull
 done
echo $n

方法三:

#!/bin/bash
sum=0
for n in $(seq 1 100);do
    i=$[$n/3]  #进行相除
    y=$[3*$i]  #进行相乘 、运算成功则成立
    if [ $y -eq $n ];then
        sum=$[$n+$sum]
    fi
done 
    echo $sum

方法四:

sum=0
for i in {1..100};do
  if [ $[$i%3] -eq 0 ];then #相除取余进行判定
  let sum+=$i  #进行加法运算
  fi 
done
echo $sum

打印九九乘法表

for (( i=1 ;i<=9;i++ ));do
    for ((n=1;n<=9;n++)) ;do
        w=0
        w=$[$n*$i]
    if [ $n -le $i  ];then
    echo -n " $i*$n=$w "
    fi
    done
echo
done

在/testdir目录下创建10个html文件,文件名格式为数字N(从1到10)加随机8个字母,如:1AbCdeFgH.html

方法一:

#!/bin/bash
cd /testdir &> /dev/null
if [ ! $(echo $?) -eq 0 ] ;then  #判断文件是否存在
  mkdir /testdir &> /dev/null
fi

for i in $(seq 1 10 );do
  w=$(cat /dev/urandom |tr -dc '0-9a-zA-Z!@_#?.,' |head -c 8)
  touch $i$w.html
done

方法二:

if ! [ -d /testdir ];then  #判断文件是否存在
 mkdir /testdir &> /dev/null  #如果没有创建文件
fi
for i in {1..10};do 
 touch /testdir/$i`cat /dev/urandom |tr -dc [:alpha:] |head -c 8`.html  #创建文件
done

探测局域网内的主机

172.18.250-254.X

能ping通显示并保存至/root/ip.log

要求并行探测提高效率。

[root@centos6mini zuoye]# cat 11.sh 
 #!/bin/bash
> /root/ip.log
for i in {250..255};do
  for n in {1..255};do
  {
  ping -c1 -w1 172.18."$i"."$n" &> /dev/null
  if [ $? -eq 0 ] ;then
  echo "ping 172.18."$i"."$n" is up" #>> /root/ip.log
  fi
  }&
  done
done

[root@centos6mini zuoye]# ./11.sh 
ping 172.18.250.14 is up
ping 172.18.250.26 is up
ping 172.18.250.71 is up
ping 172.18.250.81 is up
ping 172.18.250.112 is up
ping 172.18.250.154 is up
ping 172.18.250.164 is up
ping 172.18.250.166 is up
ping 172.18.250.183 is up
ping 172.18.250.195 is up
ping 172.18.250.212 is up
ping 172.18.250.216 is up
ping 172.18.250.224 is up
ping 172.18.250.249 is up
ping 172.18.250.250 is up
ping 172.18.251.18 is up

打印国际象棋棋盘

方法一:

[root@centos6mini zuoye]# cat qipan.sh 
#!/bin/bash
for n in {1..8};do
  for i in {1..8} ;do
    if [ $[$n%2] -eq 0  ];then
      if [ $[$i%2] -eq 0  ];then
         echo -e "33[41m 33[0mc"
      fi
      if [ $[$i%2] -ne 0  ] ;then
         echo -e "33[47m 33[0mc"
        
      fi
    else
       if [ $[$i%2] -eq 0  ];then
                 echo -e "33[47m 33[0mc"
            fi
            if [ $[$i%2] -ne 0  ] ;then
                 echo -e "33[41m 33[0mc"
            
            fi

    fi
  done
  echo 
done

方法二:

#!/bin/bash
for xline in $(seq 1 8);do  #横向
 for yline in $(seq 1 8);do  #纵向
    if [ $[$[$xline+$yline]%2] -eq 0 ];then  #判断;横向与纵向相加,最后相除是否等于0
     echo -e "33[47m 33[0mc"
    else
     echo -e "33[41m 33[0mc"
    fi
 done
 echo 
done

方法三:(放大版)

#!/bin/bash
for i in {1..8};do
    for x in {1..4};do #控制单行大小。对已有输出进行放大
        for n in {1..8};do
            a=`echo $[$[$i+$n]%2]`
            if [ $a -eq 0 ];then
                echo -e "e[43m" "" "" "" "" "" "" "" "e[0mc"
            else
                echo -e "e[42m" "" "" "" "" "" "" "" "e[0mc"
            fi
        done
        echo
    done
done

while循环

while CONDITION; do
循环体
done

CONDITION:循环控制条件;进入循环之前,先做一次判 断;每一次循环之后会再次做判断;条件为“true”,则执行 一次循环;直到条件测试状态为“false”终止循环

因此:CONDTION一般应该有循环控制变量;而此变量的值 会在循环体不断地被修正

进入条件:CONDITION为true

退出条件:CONDITION为false

“:” 和true的返回值都为真

[root@centos6mini zuoye]# true
[root@centos6mini zuoye]# echo $?
0
[root@centos6mini zuoye]# :
[root@centos6mini zuoye]# echo $?
0

判断/var/目录下所有文件的类型

[root@centos6mini app]# cat linshi 
#!/bin/bash
dc=0
lc=0
cc=0
bc=0
oc=0
zc=0

ls -l /var |grep -v total >/tmp/var_ftype.txt
while read lines
do
  ftype=`echo $lines |awk '{print $1}' |cut -c 1`
  case $ftype in
    d) dname=`echo $lines |awk '{print $9}'`; echo "$dname is a Directory" ; let dc+=1;;
    l) lname=`echo $lines |awk '{print $9}'`; echo "$lname is a Links of Soft " ;let lc+=1 ;;
    c) cname=`echo $lines |awk '{print $9}'`; echo "$cname is a Character of file" ;let cc+=1;;
    b) bname=`echo $lines |awk '{print $9}'`; echo "$bname is a Block file" ; let bc+=1;;
    -) zname=`echo $lines |awk '{print $9}'`; echo "$zname is a common file" ; let zc+=1;;
    *) echo "Others files"; let oc+=1

  esac

done </tmp/var_ftype.txt
echo '-----------------------------' 
echo -e "var目录下普通文件数量: $zcnvar目录下子目录数量:$dcnvar目录下链接文件数量:$lcnvar目录下字符类型文件数量: $ccnvar
目录下块设备文件数量:$bcn其他类型文件数量:$oc"
rm -f /tmp/var_ftype.txt


[root@centos6mini app]# ./linshi 
cache is a Directory
crash is a Directory
cvs is a Directory
db is a Directory
empty is a Directory
games is a Directory
gdm is a Directory
lib is a Directory
local is a Directory
lock is a Directory
log is a Directory
mail is a Links of Soft 
nis is a Directory
opt is a Directory
preserve is a Directory
run is a Directory
spool is a Directory
tmp is a Directory
www is a Directory
yp is a Directory
-----------------------------
var目录下普通文件数量: 0
var目录下子目录数量:19
var目录下链接文件数量:1
var目录下字符类型文件数量: 0
var
目录下块设备文件数量:0
其他类型文件数量:0

编写脚本,求100以内所有正奇数之和

方法一:

[root@centos6mini zuoye]# echo {1..100..2}
1 3 5 7 9 11 13 15 17 19 21 23 25 27 29 31 33 35 37 39 41 43 45 47 49 51 53 55 57 59 61 63 65 67 69 71 73 75 77 79 81 83 85 87 89 91 93 95 97 99
[root@centos6mini zuoye]# echo {1..100..2}|tr " " + |bc
2500

方法二:

[root@centos6mini 3.19]# ./1
2500
[root@centos6mini 3.19]# cat 1
#!/bin/bash
u=0 
for i in {1..100..2}; do
  let u+=i
done
echo $u

方法三:

[root@centos6mini 3.19]# ./1
2500
[root@centos6mini 3.19]# cat 1 
#!/bin/bash
u=0 
i=1
while [ $i -le 100 ] ;do

  let u+=i
  let i+=2
done
echo $u

编写脚本,提示请输入网络地址,如192.168.0.0,判断输入的网段中主机在线状态,并统计在线和离线主机各多少

方式一:

#!/bin/bash
w=0
a=0
y=0
for (( d=0;d<256;d++ ));do
  for (( f=1;f<255;f++ ));do
    {
    ping -c1 -w1 172.18."$d"."$f" &> /dev/null
    a=`echo $?`

    if [ $a -eq 0 ] ; then 
      echo "172.18."$d"."$f" up" >> /app/e.txt
      echo "172.18."$d"."$f" up "  
    fi
    if [ $a -ne 0 ] ; then 
            echo "172.18."$d"."$f" down " >> /app/r.txt
      echo "172.18."$d"."$f" down "         
    fi
  }&
  done
done 
w=`cat /app/e.txt |wc -l`
echo "通的 : $w"
y=`cat /app/r.txt |wc -l`
echo "不通的 : $y"
rm -rf /app/e.txt
rm -rf /app/w.txt

方式二:

#!/bin/bash
i=1
q=1
w=0
a=0
y=0
while [ $i -lt 255 ];do
  while [ $q -lt 255 ];do
    {
    ping -c1 -w1 172.18."$i"."$q" &> /dev/null
    a=`echo $?`
    if [ $a -eq 0 ] ; then 
      echo "172.18."$i"."$q"  up" >> /app/e.txt
      echo "172.18."$i"."$q"  :up"
    fi
    if [ $a -ne 0 ] ; then 
            echo "172.18."$i"."$q" down "  >> /app/w.txt
        
        echo "172.18."$i"."$q" down "

    fi
    }&
    let q=q+1

    
  done
  
  let  i=i+1
  q=1   ####将q的变量值从新赋值为1
done 
w=`cat /app/e.txt |wc -l`
echo "通的主机数为 : $w"
y=`cat /app/w.txt |wc -l`
echo "不通的主机数为 : $y"
rm -rf /app/w.txt /app/e.txt

编写脚本,打印九九乘法表

[root@centos6mini 3.19]# cat 99 
#!/bin/bash
q=1
w=1
while [ $q -le 9 ];do
  while [ $w -le 9 ];do
    e=0
    let e=q*w  
    if [ $w -le $q ];then
    echo -n " $w*$q=$e "
    fi

  let w+=1
  done
  let q+=1
  w=1
  echo
done
[root@centos6mini 3.19]# ./99
 1*1=1 
 1*2=2  2*2=4 
 1*3=3  2*3=6  3*3=9 
 1*4=4  2*4=8  3*4=12  4*4=16 
 1*5=5  2*5=10  3*5=15  4*5=20  5*5=25 
 1*6=6  2*6=12  3*6=18  4*6=24  5*6=30  6*6=36 
 1*7=7  2*7=14  3*7=21  4*7=28  5*7=35  6*7=42  7*7=49 
 1*8=8  2*8=16  3*8=24  4*8=32  5*8=40  6*8=48  7*8=56  8*8=64 
 1*9=9  2*9=18  3*9=27  4*9=36  5*9=45  6*9=54  7*9=63  8*9=72  9*9=81

4、编写脚本,利用变量RANDOM生成10个随机数字,输出这个10数字,并显示其中的最大值和最小值

[root@centos6mini 3.19]# ./4
25955
25748
16113
19729
8944
19049
16202
20067
30609
4309
最大值: 30609
最小值: 4309
[root@centos6mini 3.19]# cat 4 
#!/bin/bash
q=1
while [ $q -le 10 ] ;do
   echo $[RANDOM] &>> /app/o.txt
   #echo $[RAND0M] #每次随机都是新的值
  let q+=1  
done
cat /app/o.txt
w=`cat /app/o.txt |sort -rn |tail -n1`
e=`cat /app/o.txt |sort -rn | head -n 1`
echo "最大值: $e"
echo "最小值: $w"
rm -rf /app/o.txt

5、编写脚本,实现打印国际象棋棋盘[

[root@centos6mini 3.19]# cat 5,sh
#!/bin/bash
q=1
w=1
while [ $q -le 8 ] ;do
  while [ $w -le 8 ] ;do
    if [ $[$[$q+$w]%2] -eq 0 ] ;then  ##判断句中的计算需要$[]括起来:
      echo -e "33[47m 33[0mc"
    else
      echo -e "33[41m 33[0mc"
    fi
  
  let w+=1
  done
  let q+=1
  w=1
  echo
done

后续六个字符串:efbaf275cd、4be9c40b8b、44b2395c46、f8c8873ce0、b902c16c8b、ad865d2f63是通过对随机数变量RANDOM随机执行命令:

echo $RANDOM|md5sum|cut –c1-10

后的结果,请破解这些字符串对应的RANDOM值

方法一:

#!/bin/bash
>6
while true ;do
    while true ;do
        w=`echo $RANDOM ` #定义变量--随机数随时在变化,给他赋值一个变量,后边取值就不在变化了
        q=`echo $w|md5sum|cut -c1-10`
        if [ $q == efbaf275cd ] ;then
            echo "efbaf275cd=$w" #>>6
        elif [ $q == 4be9c40b8b ] ;then
            echo "4be9c40b8b=$w" #>>6
        elif [ $q == 44b2395c46 ] ;then
            echo "44b2395c46=$w" #>>6
        elif [ $q == f8c8873ce0 ] ;then
            echo "f8c8873ce0=$w" #>>6
        elif [ $q == b902c16c8b ] ;then
             echo "b902c16c8b=$w" #>>6
        elif [ $q == ad865d2f63 ] ;then
             echo "ad865d2f63=$w" #>>6
        fi

    done
done
[root@centos6mini ~]# cat 6 |sort|uniq
44b2395c46=9000
4be9c40b8b=12000
ad865d2f63=1000
b902c16c8b=3000
efbaf275cd=15000
f8c8873ce0=6000

方法二:

#!/bin/bash
sum=0
for i in {1..32767};do
    num=`echo $i |md5sum | cut -c1-10`
    case $num in
        efbaf275cd|4be9c40b8b|44b2395c46|f8c8873ce0|b902c16c8b|ad865d2f63)
            echo $i
            let sum++
        ;;
    esac
    if [ $sum -lt 6 ];then
        continue
    else
        break
    fi
done

$#和$*差别

[root@centos6mini 3.19]# ./linshi 1 2 2 3 4
1 2 2 3 4
2 2 3 4
2 3 4
3 4
4
[root@centos6mini 3.19]# cat linshi 
#!/bin/bash
while [ $# -gt 0 ];do  
  echo $*
  shift
done

[root@centos6mini 3.19]# ./linshi 1 2 2 3 4
5
4
3
2
1
[root@centos6mini 3.19]# cat linshi 
#!/bin/bash
while [ $# -gt 0 ];do  
  echo $#
  shift
done

使用一条命令,依次创建指定的用户。如执行createuser.sh tom bob alice harry

方法1:

[root@centos6mini 3.19]# ./linshi createuser.sh tom bob alice harry 
createuser.sh 用户创建成功
tom 用户创建成功
bob 用户创建成功
alice 用户创建成功
harry 用户创建成功
[root@centos6mini 3.19]# cat linshi 
#!/bin/bash
#!/bin/bash
while [ $# -gt 0 ]
do 
    

    id $1 &>/dev/null
    if [ $? -eq 0 ];then
        echo $1 is already exsit.
        shift
        continue
    fi
    useradd $1
    echo "$1 用户创建成功"
    shift

done

删除用户
[root@centos6mini 3.19]# ./linshi createuser.sh tom bob alice harry 
createuser.sh 用户删除成功
tom 用户删除成功
bob 用户删除成功
alice 用户删除成功
harry 用户删除成功
[root@centos6mini 3.19]# cat linshi 
#!/bin/bash
while [ $# -gt 0 ];do 
      userdel -r $1
      echo "$1 用户删除成功"
      shift
done

方法2:

#!/bin/bash
while [ $# -gt 0 ];do
    id $1 &> /dev/null && { echo $1 is alread exist && shift && continue; }
    useradd $1 && echo $1 is created
    shift
done

每隔3秒钟到系统上获取已经登录的用户的信息;如果发现用户hacker登录,则将登录时间和主机记录于日志/var/log/login.log中,并退出脚本

#!/bin/bash
while true;do
if $(ps -au |grep "^hacker.*" &> /dev/null);then  #监测前台终端上有没有用户登陆信息
    echo $(ps -au |grep "^hacker.*" |tr -s " " |cut -d" " -f1,7,9) >> /var/log/login.log
      #将用户登陆信息保存到日志文件中/var/log/login.log
    break  #推出监测
else
    sleep 3s  #三秒已监测,防止程序太多占用cpu
fi
done

随机生成10以内的数字,实现猜字游戏,提示比较大或小,相等则退出

[root@centos6mini 3.19]# ./linshi 
请输入数值 : 1
错了 ,小了
请输入数值 : 9
错误,大了
请输入数值 : 5
猜对了
[root@centos6mini 3.19]# cat linshi 
#!/bin/bash
qe=$( echo $[$RANDOM%10] ) 
while true ;do
read -p "请输入数值 : " qw
    if [ $qw -gt $qe  ] ;then
        echo "错误,大了"

    elif [ $qw -lt $qe  ] ;then
        echo "错了 ,小了"
    elif [ $qe -eq $qw ];then
        echo "猜对了"
        break
    fi

done

用文件名做为参数,统计所有参数文件的总行数

[root@centos6mini 3.19]# ./linshi /etc/passwd /etc/sysconfig/network-scripts/ifcfg-eth1 /etc/passwd
89
[root@centos6mini 3.19]# cat linshi 
#!/bin/bash
sumhang=0
while [ $# -gt 0 ];do
    hang=$( cat $1 | wc -l )
    let sumhang+=hang
    shift
done
echo $sumhang

4、用二个以上的数字为参数,显示其中的最大值和最小值

方法一:

#!/bin/bash
min=$1 #最小值等于第一个数值
max=$1 #最大数等于第一个数值

while [ $# -gt 0 ];do  #进行数值数量进行判断,做循环
    value=$1   #定义变量
    if [ $value -gt $max ];then  #当第二个数大于第一个数时
        max=$value #最大值等于后一个数值,对max从新赋与新的值
    elif [ $value -lt $min ];then  #当第二个数大于第一个数时
        min=$value  #最小值等于后一个数值 对min从新赋与新的值
    fi
    shift #不管前面判断是否成立,都进行数列左移,
done
echo "min $min"
echo "max $max"

方法二:

#!/bin/bash
if [ $# -lt 2 ];then #对输入数字数量进行检测
    echo "type more than two numbers"
    exit 1
fi

small=$(echo $* |xargs -n1 |sort -n |head -n 1)  #进行数值排序
big=$(echo $* |xargs -n1 |sort -n |tail -n 1)
echo "Maximum number is $big"
echo "Minimum number is $small"

[root@localhost ~]# echo 1 -3 -99 46 7 32 43 2133 312 |xargs -n1 #以一数列进行显示
1
-3
-99
46
7
32
43
2133
312

read 示例:

[root@centos6mini 3.19]# while read -p "type your name: " name ;do echo your is $n$name; done
type your name: zhangsan
your is zhangsan
type your name: lisi
your is lisi
type your name: wangwu
your is wangwu
type your name: ^C

[root@centos6mini 3.19]# read bing < test。txt
[root@centos6mini 3.19]# echo $bing  
hello

^C
[root@centos6mini 3.19]# while read name ; do echo $name ;done # 没有指定输入
123   #输入
123   #输出
qwe    #输入
qwe   #输出
asd   #输入
asd   #输出
zxc   #输入
zxc   #输出
^C^[[A^C
[root@centos6mini 3.19]# while read name ; do echo $name ;done <6  #name为变量,输入来自文件文件内容读取完了,循环停止
hello
qweqew
12341234
asdas
ads
sda
sad
asd
as
asd
as

read 示例:
显示UID小于1000的为sys user,大于等于1000的为comm user。

[root@centos6mini 3.19]# cat linshi    
#!/bin/bash
while read usline ;do #通过read ,定义变量值usline
    usuid=$( echo $usline |cut -d: -f3 ) ,#引用变量usline ,并定义变量usuid
    usname=$( echo $usline |cut -d: -f1 ) #引用变量usline ,并定义变量usname
    #if [ $usuid -lt 1000 ];then  #进行判断
    if (( $usuid < 1000 )) ;then  #进行判断(两种方式均可以)
        echo "$usname is a sys user"
    else 
        echo "$usname is a comm user"
    fi

done < /etc/passwd #每次循环引用文件内容一行,进行操作,引用完了,循环停止。
[root@centos6mini 3.19]# ./linshi  
root is a sys user
bin is a sys user
daemon is a sys user
adm is a sys user
lp is a sys user
sync is a sys user
shutdown is a sys user
halt is a sys user
mail is a sys user
uucp is a sys user
operator is a sys user
games is a sys user
gopher is a sys user
ftp is a sys user
nobody is a sys user
vcsa is a sys user
saslauth is a sys user

找出分区利用率大于10%的分区,并发出警告/dev/sdaX will be full: 15%要求使用while read来实现

答案1:

#!/bin/bash
df |grep /dev/sd > /app/fenqu
while read line;do
    used=$(echo $line |tr -s " " % |cut -d% -f5)
    name=$(echo $line |cut -d" " -f1)
    if (( $used > 10 ));then
        echo "$name will be full:$used%"
    fi
done < /app/fenqu


[root@centos6mini 3.19]# df |grep /dev/sd >/app/zuoye/3.19/6
[root@centos6mini 3.19]# cat /app/zuoye/3.19/6
/dev/sda5   28564480 1872060 25234756 7% /
/dev/sda2   20511356 70276 19392504 1% /app
/dev/sda1    194241 31785  152216 18% /boot
[root@centos6mini 3.19]# while read line ;do
> used=$(echo $line |tr -s " " % |cut -d% -f5)
> name=$(echo $line |cut -d" " -f1)
> if (( $used > 10 ));then
> echo "$name will be full:$used%"
> fi
> done </app/zuoye/3.19/6
/dev/sda1 will be full:18%

答案2:

#!/bin/bash
> df_h
df -h | grep ^/dev/[sh]d* >> df_h
while read disk;do
  a=`echo $disk |grep -o "<[0-9]{0,3}%" | cut -d% -f1`
  b=`echo $disk |grep -o "^[^[:space:]]+>"`
  if [ $a -gt 10 ];then
    echo "$b will be full: $a%"
  fi 
done < df_h

答案3:sed实现

#!/bin/bash
df |grep /dev/sd |while read disk;do
    diskused=$(echo $disk |sed -r 's/.* ([0-9]+)%.*/1/')
    diskname=$(echo $disk |sed -r 's@(/dev/sd[[:lower:]][[:digit:]]).*@1@')
    if (( $diskused > 10 ));then
        echo "$diskname will be full:$diskused%"
    fi
done

until循环

until CONDITION; do
循环体
done

进入条件: CONDITION 为false

退出条件: CONDITION 为true

循环控制语句continue
用于循环体中

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

while CONDTIITON1; do
      CMD1
       ...
       if CONDITION2; then
        continue
         fi
       CMDn

循环控制语句break

用于循环体中

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

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

循环控制shift命令

shift [n]

用于将参量列表 list 左移指定次数,缺省为左移一次。

参量列表 list 一旦被移动,最左端的那个参数就从列表中删 除。while 循环遍历位置参量列表时,常用到 shift

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

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

特殊用法

1、while循环的特殊用法(遍历文件的每一行):

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

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

2、双小括号方法,即((…))格式,也可以用于算术运算

双小括号方法也可以使bash Shell实现C语言风格的变量操作

I=10

((I++))

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

控制变量初始化:仅在运行到循环代码段时执行一次

控制变量的修正表达式:每轮循环结束会先进行控制变量修 正运算,而后再做条件判断

select循环与菜单

select variable in list
    do
        循环体命令
    done

select 循环主要用于创建菜单,按数字顺序排列的 菜单项将显示在标准错误上,并显示 PS3 提示符, 等待用户输入
用户输入菜单列表中的某个数字,执行相应的命令
用户输入被保存在内置变量 REPLY 中
select与case

select 是个无限循环,因此要记住用 break 命令退 出循环,或用 exit 命令终止脚本。也可以按 ctrl+c 退出循环
select 经常和 case 联合使用
与 for 循环类似,可以省略 in list,此时使用位置 参量

信号捕捉trap

trap '触发指令' 信号
            自定义进程收到系统发出的指定信号后,将执行触发指令 ,而不会执行原操作
trap '' 信号
            忽略信号的操作
trap '-' 信号
            恢复原信号的操作
trap -p
            列出自定义信号操作

文件测试

-z string 测试指定字符是否为空,空着真,非空为假
-n string 测试指定字符串是否为不空,空为假 非空为真 -e FILE 测试文件是否存在 -f file 测试文件是否为普通文件 -d file 测试指定路径是否为目录
-r file 测试文件对当前用户是否可读
-w file 测试文件对当前用户是否可写
-x file 测试文件对当前用户是都可执行
-z 是否为空 为空则为真 -a 是否不空
-u file :是否存在且拥有suid权限
-g file :是否存在且拥有sgid权限
-k file :是否存在且拥有sticky权限

[root@centos6mini 3.19]# [ -n "" ]&& echo true #空为假
[root@centos6mini 3.19]# [ -n "1" ]&& echo true #非空为真
true
[root@centos6mini 3.19]# [ -n " " ]&& echo true #非空为真
true
[root@centos6mini 3.19]# [ -n ]&& echo true #非空为真
true
[root@centos6mini 3.19]# [ -z ]&& echo true #空为真
true
[root@centos6mini 3.19]# [ -z "" ]&&echo true #空为真
true
[root@centos6mini 3.19]# [ -z "1" ] && echo true "#非空为假
[root@centos6mini 3.19]# [ -z " " ] && echo true #非空为假

扫描/etc/passwd文件每一行,如发现GECOS字段为空,则填充用户名和单位电话为62985600,并提示该用户的GECOS信息修改成功。

#!/bin/bash
cat /etc/passwd |while read userline;do #用管道实现给循环while赋值
    if [ -n "$(echo $userline |cut -d: -f5)" ];then  #-n "1" 非空为真
        continue   #为真时啥都不做
    else
        username=$(echo $userline|cut -d: -f1)
        usermod -c $username-62985600 $username
        echo $username's gecos already changed!
    fi
done

特殊用法

双小括号方法,即((…))格式,也可以用于算术运算

双小括号方法也可以使bash Shell实现C语言风格的变量操作

I=10

((I++))

for循环的特殊格式:

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

do

循环体

done

控制变量初始化:仅在运行到循环代码段时执行一次

控制变量的修正表达式:每轮循环结束会先进行控制变量修 正运算,而后再做条件判断

练习:计算从1加到100的和。

#!/bin/bash
sum=0
for ((i=1;i<=100;i++));do
    let sum+=i
done
echo sum=$sum
unset sum i
 
sum=0
i=1
while [ $i -le 100 ];do
    let sum+=i
    let i++
done
echo sum=$sum

select循环与菜单
select variable in list

do

循环体命令

done

select 循环主要用于创建菜单,按数字顺序排列的 菜单项将显示在标准错误上,并显示 PS3 提示符, 等待用户输入

PS3="Please choose your menu: "

用户输入菜单列表中的某个数字,执行相应的命令

用户输入被保存在内置变量 REPLY 中

[root@centos6mini 3.19]# ./linshi   
1) huangmenji
2) huimian
3) hulatang
4) roujiamo
Please choose your menu: 1
The price is $15
Please choose your menu: 2
The price is $10
Please choose your menu: 3
The price is $8
Please choose your menu: 4
The price is $6
Please choose your menu: 5
get out

[root@centos6mini 3.19]# cat ./linshi  
#!/bin/bash

select menu in huangmenji huimian hulatang roujiamo;do 
    case $REPLY in
        1) echo "The price is $15";;
        2) echo "The price is $10";;
        3) echo "The price is $8";;
        4) echo "The price is $6";;
        *) echo "get out";break;;
    esac
done

信号捕捉trap

trap '触发指令' 信号

自定义进程收到系统发出的指定信号后,将执行触发指令 ,而不会执行原操作

trap '' 信号 忽略信号的操作

rap '-' 信号 恢复原信号的操作

trap -p 列出自定义信号操作

trap 所有可选项

[root@centos6mini 3.19]# trap -l 
 1) SIGHUP    2) SIGINT    3) SIGQUIT   4) SIGILL    5) SIGTRAP
 6) SIGABRT   7) SIGBUS    8) SIGFPE    9) SIGKILL   10) SIGUSR1
11) SIGSEGV   12) SIGUSR2   13) SIGPIPE   14) SIGALRM   15) SIGTERM
16) SIGSTKFLT  17) SIGCHLD   18) SIGCONT   19) SIGSTOP   20) SIGTSTP
21) SIGTTIN   22) SIGTTOU   23) SIGURG   24) SIGXCPU   25) SIGXFSZ
26) SIGVTALRM  27) SIGPROF   28) SIGWINCH  29) SIGIO    30) SIGPWR
31) SIGSYS   34) SIGRTMIN  35) SIGRTMIN+1 36) SIGRTMIN+2 37) SIGRTMIN+3
38) SIGRTMIN+4 39) SIGRTMIN+5 40) SIGRTMIN+6 41) SIGRTMIN+7 42) SIGRTMIN+8
43) SIGRTMIN+9 44) SIGRTMIN+10 45) SIGRTMIN+11 46) SIGRTMIN+12 47) SIGRTMIN+13
48) SIGRTMIN+14 49) SIGRTMIN+15 50) SIGRTMAX-14 51) SIGRTMAX-13 52) SIGRTMAX-12
53) SIGRTMAX-11 54) SIGRTMAX-10 55) SIGRTMAX-9 56) SIGRTMAX-8 57) SIGRTMAX-7
58) SIGRTMAX-6 59) SIGRTMAX-5 60) SIGRTMAX-4 61) SIGRTMAX-3 62) SIGRTMAX-2
63) SIGRTMAX-1 64) SIGRTMAX

#!/bin/bash 
trap 'echo “signal:SIGINT"' int   # int 相当与Ctrl+c
trap -p 
for((i=0;i<=10;i++)) 
do 
sleep 1
echo $i 
done
trap '' int #trap '' 信号 忽略信号的操作 
trap -p 
for((i=11;i<=20;i++)
do 
sleep 1
echo $i 
done 
trap '-' int  #rap '-' 信号 恢复原信号的操作 
trap -p 
for((i=21;i<=30;i++)
do 
sleep 1 
echo $i
done

忽略kill -15 ,操作可以成功

忽略kill -9操作 ,不成功,无法忽略kill -9

trap -l,把所有信号打印出来

查看kill -9 和kill -15 对应的信号选项

select与case

select 是个无限循环,因此要记住用 break 命令退 出循环,或用 exit 命令终止脚本。也可以按 ctrl+c 退出循环

select 经常和 case 联合使用

与 for 循环类似,可以省略 in list,此时使用位置