前言:

       在Linux 系统管理的过程中,能够熟练使用shell中的重要工具bash进行编程,在如今运维自动化的趋势中尤为重要。

        与其他编程语言相比bash没有属于自己的函数库,bash在实现功能时都是通过PATH路径调用外部的命令,$PATH 路径包含的命令决定了bash可以实现的功能。$PATH路径就是bash的函数库。也就是说bash脚本的编程更像是命令的堆砌,这使得bash编程相对来说更简洁易懂,尤其擅长处理系统管理方面的工作。我们只要熟悉了bash基本特性,就能够写出能够执行复杂任务的高效脚本,在运维工作中,很多问题将会变得事半功倍。本文主要介绍shell编程控制语句并附加相关示例来加深理解与记忆。相关内容如下,请读者自行参阅:

1、if语句

2、case语句

3、for循环及其特殊用法

4、while循环及其特殊用法

5、until循环

6、循环控制语句continue

7、循环控制语句break

8、循环控制shift命令

9、select循环与菜单

10、信号捕捉trap



1、条件选择:if语句

    ①单分支

        if 判断条件;then
            条件为真的分支代码
        fi

    ②双分支        

        if 判断条件; then
              条件为真的分支代码
        else
              条件为假的分支代码
        fi

    ③多分支      

         if 判断条件1; then
               条件为真的分支代码
         elif 判断条件2; then
               条件为真的分支代码
         elif 判断条件3; then
               条件为真的分支代码
         else
               以上条件都为假的分支代码
         fi

示例:请用户输入成绩:80-100为优秀,60-79为通过,其他淘汰。

 #!/bin/bash
 read -p "pelease input your score: " score
 if [ "$score" -ge 80 -a "$score" -le 100 ];then
       echo "优秀"
 elif [ "$score" -lt 80 -a "$score" -ge 60 ];then
       echo "通过"
 else
       echo "淘汰"
 fi

2、条件判断: case语句

 语法:case 变量引用 in
    PAT1)
         分支1;;
    PAT2)
        分支2;;
    ...
      *)
       默认分支;;
        esac

       case支持glob风格的通配符
              *: 任意长度任意字符
              ?: 任意单个字符
              []:指定范围内的任意单个字符
              a|b: a或b

              。。。

      示例:编写脚本,提示用户输入yes或no不区分大小写,并判断用户输入的是yes还是no或者其他信息。

#!/bin/bash
read -p "please input yes or no: " yn
ans=$(echo $yn|tr "[[:upper:]]"  "[[:lower:]]")
case $ans in
y|yes)
   echo yes;;
n|no)
   echo no ;;
*)
  echo yes/no?
esac

3、for循环

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

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

     列表生成方式:
      (1) 直接给出列表
      (2) 整数列表:
            (a) {start..end}
            (b) $(seq [start [step]] end)
      (3) 返回列表的命令
            $(COMMAND)
      (4) 使用glob, 如: *.sh
      (5) 变量引用;
            $@, $*

    示例:用for循环打印九九乘法表。

#!/bin/bash
for i in {1..9};do
    for j in $(seq $i);do
       echo -ne "$j*$i=$[$i*$j]\t"
    done
    echo
done

       for循环的特殊格式:     

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

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

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

    示例:用for循环特殊格式打印九九乘法表。


#!/bin/bash
for ((i=1;i<=9;i++)); do

    for ((j=1;j<=i;j++)); do
       echo -ne "$j*$i=$[$j*$i]\t"
    done
    echo
done

4、while循环

    语法:while CONDITION; do
                        循环体
               done

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

示例:求100以内能被3整除的数之和

#!/bin/bash
declare i=1
declare sum=0
while [  "$i" -lt 100 ];do
      let i++ 
      if [ $[i%3] -ne 0 ];then
         continue
      fi  
      let sum+=$i
done
echo sum is $sum

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

       while read line; do
                循环体
       done < /PATH/FROM/SOMEFILE
       依次读取/PATH/FROM/SOMEFILE文件中的每一行,且将行赋值给变量line(也可以是其他自定义变量名)。

       示例:找出ID号为偶数的所有用户,显示其用户名及ID号。

while read line ;do
     if [ $[`echo $line|cut -d: -f3`%2] -eq 0 ];then
                echo -ne "username:`echo $line|cut -d: -f1` \t"
                echo "uid:`echo $line|cut -d: -f3`"
     fi  
done 
  

       或者将内容通过管道传给while循环:

       示例:若磁盘利用率大于80%,则通知用户。

#!/bin/bash
df|grep "/dev/sd"|while read disk ;do
      diskused=$(echo $disk|sed -r 's/.* ([0-9]+)%.*/\1/')
      diskname=$(echo $disk|cut -d" " -f1)
      [ $diskused -ge 80 ] && echo "$diskname will bi full: $diskused%"
done


5、untill循环

     语法until CONDITION; do
                    循环体

           done
           进入条件: CONDITION 为false
           退出条件: CONDITION 为true

示例:用until打印九九乘法表

#!/bin/bash
j=1
i=1
until [ "$j" -gt 9 ];do
     until [ "$i" -gt "$j" ];do
        echo -ne "$i*$j=$[i*j]\t"
        let i++
     done
     echo
     let i=1
     let j++
done

6、循环控制语句continue

      continue用于循环体中
      continue [N]:提前结束第N层的本轮循环,而直接进入下一轮判断;最内层为第1层
      while CONDTIITON1; do
              CMD1
              ...
      if CONDITION2; then
      continue
      fi
      CMDn
       ...
      done

示例:打印除了5以外1-10之间的数字。

#!/bin/bash
for i in {1..10};do
    if [ $i -eq 5 ];then
        continue
    else
        echo $i
    fi                                                                                                                                                 
done

7、循环控制语句break

      语法:

      break [N]:, 最内层为第1层
      while CONDTIITON1; do
               CMD1
               ...
          if CONDITION2; then
               break
          fi
          CMDn
          ...
      done

      break与continue的区别在于,continue是在循环中跳过满足条件的单次循环,而break则默认跳出自己处于的循环体。continue示例中如果换为break则只打印1-4,由于5触发break跳出循环之后的不在打印。

8、循环控制shift命令

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

#!/bin/bash
while (($#>0));do
   echo $*
   shift
done 
显示结果:
[root@R1 app]# ./haha.sh a b c d e
a b c d e
b c d e
c d e
d e
e

9、select循环与菜单

select variable in list
do
循环体命令
done
  • select 循环主要用于创建菜单,按数字顺序排列的菜单项将显示在标准错误上,并显示PS3提示符,等待用户输入。

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

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

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

  • select 经常和 case 联合使用

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


    示例:

#!/bin/bash
PS3="please choose your menu: "
select menu in huimian lamian hulatang yrt;do
      case $REPLY in                                                                                                                                   
      1)
          echo "the price is \$10"   ;;
      2)
          echo "the price is \$15"   ;;
      3)
          echo "the price is \$5"    ;;
      4)
          echo "the price is \$20"   ;;
      *)
          echo "get out!"
          break
      esac
done

10、信号捕捉trap

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

示例:

#!/bin/bash
 trap 'echo “signal:SIGINT"' int
 trap -p
 for((i=0;i<=10;i++))
 do
 sleep 1
 echo $i
 done
 trap '' int
 trap -p
 for((i=11;i<=20;i++))
 do
 sleep 1
 echo $i
 done
 trap '-' int
 trap -p
 for((i=21;i<=30;i++))
 do
 sleep 1
 echo $i
 done