bash脚本编程之判断和循环

bash脚本编程之判断和循环

     变量:
      特性:一个变量中只能存储一个数值;

     数组:能够容纳多个数组元素的连续的内存空间;

      1)稀疏数组:
      2)稠密数组: 

    数组元素:数组中任何一个存放数据的存储单元,其作用相当于一个变量;

    数组元素的标识:

      索引数组标识:所有的数组元素都是使用数字编号的;
      通常数字标号是从0开始的,即:0,1,2....

    关联数组标识:所有的数组元素都可以使用名称(字符串)来标识;
      注意:bash4.0以上版本才有可能支持关联数组;

   数组声明定义:
      1.declare命令:
         -a to make NAMEs indexed arrays (if supported)
           将其后的变量名称声明为索引数组;declare -a names=([0]='z'[1]='w' [2]='1')
         -A to make NAMEs associative arrays (if supported)
           将其后的变量名称声明为关联数组;declare -a names=( "z" "w" "l" )

     2.直接使用变量赋值的方式:
         定义稠密的索引数组
         ARRAY-NAME=(“VALUE1” “VALUE2”....)

         定义稀疏数组:
          ARRAY-NAME=([0]=“VALUE1” [1]=“VALUE2”....)

         定义关联数组:
           ARRAY-NAME=(INDEX_name1='VALUE1' INDEX_VALUE2='VALUE2'....)

     3.分别定义数组元素:
         ARRAY_NAME[0]='VALUE' 
         ARRAY_NAME[1]='VALUE'      
         ARRAY_NAME[2]='VALUE'
         .......

  引用数组元素的方式:
        ${ARRAY_NAME[INDEX]} 
        注意:在引用数组元素时,没人给出索引号,默认编号为”0“,即显示第一个数组元素的数值;

  引用整个数组中所有元素:
        ${ARRAY_NAME[*]}或 ${ARRAY_NAME[@]} 

  引用整个数组的所有元素的索引号:
        ${!ARRAY_NAME[*]}或 ${!ARRAY_NAME[@]}

  查看数组中的元素个数(数组长度):
       ${#ARRAY_NAME[*]}或 ${#ARRAY_NAME[@]}

  数组切片:
        ${ARRAY_NAME[*]:offset}
          //显示包括offste数值所对表示的位置的元素极其后所有的元素;

        ${ARRAY_NAME[*]:offset:number}
          //显示数组中包括offset数值所对应表示的位置的元素及其后number个元素;

  数组撤销:
        unset ARRAY_NAME

  RANDOM变量:
        随机数变量:0-32767,整数值;

        从熵池中取随机数;

        熵池:
        /dev/random
          两次敲击键盘的时间间隔;
          两次IO的时间间隔;
          ...
        /dev/urandom:伪熵池
          利用应用程序计算得到的随机数;
                        [root@localhost ~]# echo $RANDOM
         25570
         [root@localhost ~]# echo $RANDOM
         8882
         [root@localhost ~]# echo $RANDOM
         2373
        [root@localhost ~]# echo $RANDOM
        4870
       [root@localhost ~]# echo $RANDOM
        3895
       [root@localhost ~]# echo $RANDOM
        8353
      [root@localhost ~]# echo $RANDOM
       14099

 bash脚本编程的结构:
   bash脚本编程语言:
        脚本类语言
        解释型语言
        过程式编程语言

      过程式编程语言结构:
       顺序执行结构:默认
         从上而下,自左而右地执行所有的语句(命令);

       选择执行结构:
         当条件满足或不足时,才会执行对应的语句(命令);

       循环执行结构:
         重复执行某段语句(命令);    

   bash脚本编程语中也具有上述结构;
        顺序执行结构:默认

       选择执行结构:
        根据给定条件逻辑判断结果或根据某个可选取的取值范围,进而选择某个分支结构中的命令语句予以执行的方式;
       if:
         选择执行结构的标准,根据条件的逻辑判断结果选择执行的语句内容;
       case:
         选择执行结构的标准,根据符合某特定范围的取值标准选择执行的语句内容;

       循环执行结构:
        对于特定语句内容 ,重复执行0,1,多次;

       for:以遍历列表的方式进行循环;
       while:根据给定条件的逻辑判断结果进行循环,逻辑判断结果为真,循环,否则,停止循环;   
       until:根据给定条件的逻辑判断结果进行循环,逻辑判断结果为假,循环,否则,停止循环;
       select:死循环,即没有默认退出条件的循环;利用循环提供一个可选择的列表;

 bash脚本的执行结构之if选择执行结构:
      if语句:if - if COMMANDS; then COMMANDS; [ elif COMMANDS; then COMMANDS; ]... [ else COMMANDS; ] fi

    if语句的单分支结构:
        if - if COMMANDS; then COMMANDS; fi

        注意:是否会执行then后面的命令,取决于if后面的命令的执行状态返回值:
            1.如果其返回值为真,则执行then后面的命令;
            2.如果其返回值为假,则不执行then后面的命令;

      建议在脚本中的书写格式:
       if CONDITION ; then
         statement
          ...
       fi 

       或

       if  CONDITION

       then

         STATEMENT
         ....
       fi

       if语句的双分支结构:
        if - if COMMANDS; then COMMANDS; else COMMANDS; fi

        注意:是否执行then后的命令或else后的命令,取决于if后的命令执行状态的返回值;
        1.如果返回值为真,则执行then后的命令;
        2.如果为假,则执行else后的命令;

       if语句的多分支结构:

         if - if COMMANDS; then COMMANDS; else COMMANDS; fi

        注意:是否执行then后的命令或else后的命令,取决于if后的命令执行状态的返回值或else后的命令执行状态返回值;

        1.首先判断if后的命令的状态返回值是否为真,为真就执行then后的语句;如果为假,就继续判断第一个elif后的命令
          执行状态返回值;
        2.第一个elif后的命令执行状态返回值为真,就执行第一个elif语句中then后的命令,否则,就继续判断第二个elif后的
          执行状态返回值;
       3.以此类推,会判断每个elif后面的命令的执行状态返回值是否为真;如果所有的if和elif后面的命令的执行状态返回值均为假,则执行else后面的语句;

        建议在脚本中的书写格式:
            if CONDITION1 ; then
                STATEMENT
                ...
            elif CONDITION2 ; then
                STATEMENT
                ...
            elif CONDITION3 ; then
                STATEMENT
                ...
            ...
            else
                STATEMENT
                ...
            fi
            或
            if CONDITION
            then
                STATEMENT
                ...
            elif CONDITION2 ; then
                STATEMENT
                ...
            elif CONDITION3 ; then
                STATEMENT
                ...
            ...
            else
                STATEMENT
                ...
            fi

       注意:if的多分支结构,使用场景不多,而且有时候,可以使用嵌套的多分支或双分支if结构代替if多分支结构;

       嵌套的if结构
         if CONDITION1 ; then
            if CONDITION2 ; then
               if CONDITION3 ; then
                STATEMENT
                ...

               else
                STATEMENT
                ...

            else
             STATEMENT
              ...
              fi

         else
          STATEMENT
          ...
          fi

     COMMAND1 && COMMAND2 || COMMAND3

    示例:
      写脚本,判断某个用户的默认登录shell是否为/bin/bash

      #!/bin/bash
      y=$(cut -d: -f 1 /etc/shadow | sort -R | head -1)
      w=$(egrep "^$y\>" /etc/passwd | cut -d: -f 7)
      if [ "$w"=="/bin/bash" ] ; then
          echo "${y}'s login shell is /bin/bash."
      else
          echo "${y}'s login shell is ${w}"
      fi

      unset y w

     bash脚本编程之用户交互使用:
       位置参数变量:
       $0:命令的本身,对于脚本而言,就是该脚本的路径;
       $1,$2,....$N:脚本后面通过命令行给脚本传递的命令行参数;
         N>9时,引用该位置变量时需要加{},即${10}

      特殊变量:
      $@:给出的所有位置的参数的列表,当使用双引号时,每个参数作为单独的字符串存在;
      $*:给出的所有位置的参数的列表,当使用双引号时,每个参数作为单独的字符串存在;
      $#:表示除去$0之外,整个命令行中有多少个参数;

     read命令:
       read [-ers] [-a 数组] [-d 分隔符] [-i 缓冲区文字] [-n 读取字符数] 
            [-N 读取字符数] [-p 提示符] [-t 超时] [-u 文件描述符] [名称 ...]

        -a array:定义数组(索引数组);
        -p prompt:给用户输出提示信息;
        -t timeout:用户输入的超时时间;
         name :变量或数组的名称;如果省略此内容,bash会将read读到的信息直接保存到内置的名为REPLAY变量中;

     注意:
        Linux思想之一:尽量不与用户交互;

        在使用read命令时,通常会使用-t选项来指定与用户的交互时间,一旦超过预定时间,脚本中后续的命令内容
        会自动被执行;因此,通常需要在后面判断通过read赋值的变量值是否为空,如果为空,可能为该变量提供默认值;

        read -t 5 VAR1
          [ -z $VAR1 ] && VAR1=value1

   管理用户脚本:
   脚本可以接受两个参数,第一个参数为-a或-d,第二个参数为用户名;如果第一个参数是-a,则创建其后面参数命名的用户;如果第一个参数为-d,则删除其后面参       数命名的用户;

     #!/bin/bash
     #
    if [ $# -ne 2 ] ; then
    echo "Make sure provide TWO arguments."
      exit 5
      fi

    if [ $1 == '-a' ] ; then
      if ! id $2 &> /dev/null ; then
        useradd $2 &> /dev/null
        echo $2 | passwd --stdin $2 &> /dev/null
        echo "User $2 created succesfully and password changed to it's username."
      else
        echo "$2 exists already."
      fi
    elif [ $1 == '-d' ] ; then
      if id $2 &> /dev/null ; then
        userdel -r $2 &> /dev/null
        echo "User $2 delete finished."
      else
        echo "$2 does not exist yet."
      fi
      else
        echo "Usage: $(basename $0) -a USERNAME | -d USERNAME"
      exit 6
    fi

改进版:使用read命令;
#!/bin/bash
#
if [ $# -ne 1 ] ; then
echo "Make sure provide ONE argument."
exit 5
fi

if [ $1 == '-a' ] ; then
read -p "Please input a username for creating: " USERNAME
if ! id $USERNAME &> /dev/null ; then
useradd $USERNAME &> /dev/null
echo $USERNAME | passwd --stdin $USERNAME &> /dev/null
echo "User $USERNAME created succesfully and password changed to it's username."
else
echo "$USERNAME exists already."
fi
elif [ $1 == '-d' ] ; then
read -p "Please input a username for deleting: " USERNAME
read -t 5 -p "confirm? Input 'yes' to continue: " CHOICE
[ -z $CHOICE ] && CHOICE='no'

if [ $CHOICE == 'yes' ] ; then
if id $USERNAME &> /dev/null ; then
userdel -r $USERNAME &> /dev/null
echo "User $USERNAME delete finished."
else
echo "$USERNAME does not exist yet."
fi
else
echo
echo "$USERNAME is not deleted."
fi
else
echo "Usage: $(basename $0) { -a | -d }"
exit 6
fi

写脚本解决问题:
1.判断用户通过命令行给的一个参数是否为整数。

#!/bin/bash

if [ $# -ne 1 ] ; then
echo "Make sure provide ONE digit."
exit 5
fi

if [[ $1 =~ ^[[:digit:]]+$ ]] ; then
echo "$1 is a pure digit."
else
echo "$1 is not a digit."
fi

    循环执行结构:
       循环:将某一段代码或命令重复执行0,1或多次;

       一个好的循环结构,必须要包括两个重要的环节:
        1.进入循环的条件:
          在符合要求或满足条件时才开始循环;

        2.退出循环条件:
          达到某个要求或符号某个条件时需要结束或终止循环的执行;

    for循环:
       1.遍历列表的循环:
         for 名称 [in 词语 ... ] ; do 命令; done
         为列表中的每个成员执行命令。

         建议在脚本中的书写格式:
         for VAR_NAME in LIST ; do
           循环体
         done
         或
         for VAR_NAME in LIST 
         do
            循环体
         done

         注意:
         VAR_NAME:任意指定的变量名称,变量的值是从LIST中遍历获取的各个元素;
         LIST:for循环需要遍历的列表,可以通过如下方式生成列表:
            1.直接给出列表;
            2.纯整数列表:
              1):花括号展开:
                   {FIRSTNUM..LASTNUM}

              2):seq命令
                    seq [OPTION]... LAST  seq 10

                    seq [OPTION]... FIRST LAST  seq 2 6

                    seq [OPTION]... FIRST INCREMENT LAST   seq 0 2 20

           3.花括号展开
                   {FIRST..LAST}

           4.命令的执行结果:
             ls /etc

           5.GLOBBING通配符

           6.某些特殊变量的值:
             $*,$@

          循环体:

            一般来说,循环体中应该包括能够用到VAR_NAME变量的值的命令或命令的组合;如果循环体中的命令没有用到
            VAR_NAME变量的值的话,列表的元素的个数就是此次for循环的次数;

            #!/bin/bash
            for i in {1..50} ; do 
            sum=$[sum+i]
            done
            echo "$sum" 

       例子:给脚本传递三个整数,要求:
1) 如果用户传递过来的不是三个参数,报告正确用法;
2) 如果三个参数中有非纯数字字符串,报告错误并提示用户输入数字;
3) 从三个整数中的选出最大数和最小数,并显示出来;
4) 不能使用sort命令等排序;
[root@localhost ~]# vim a.sh
#!/bin/bash
#
if [ $# -ne 3 ] ; then
   echo " please provide three argument"
   exit 5
fi
if [[ $1 =~ [^[:digit:]] ]]||[[ $2 =~ [^[:digit:]] ]] ||[[ $3 =~ [^[:digit:]] ]] ; then
   echo "exist error"
fi
if [ $1 -ge $2 ] ; then
    if [ $1 -ge $3 ] ; then
       echo "the largest is $1"
    fi
fi
if [ $2 -ge $1 ] ; then
    if [ $1 -ge $3 ] ; then
      echo "the largest is $2"
    fi
fi
if [ $3 -ge $1 ] ; then
   if [ $3 -ge $2 ] ; then
     echo "the largest is $3"
   fi
fi
if [ $1 -le $2 ] ; then
   if [ $1 -le $3 ] ; then
     echo "the smallest is $1"
   fi
fi
if [ $2 -le $1 ] ; then
   if [ $2 -le $3 ] ; then
     echo "the smallest is $3"
   fi
fi
if [ $3 -le $1 ] ; then
  if [ $3 -le $2 ] ; then
    echo "the largest is $3"
  fi
fi

结果:
 [root@localhost ~]# bash a.sh  12  13 14
 the largest is 14
 the smallest is 12

例:给脚本传递一个整数,分别计算该整数以内所有偶数之和以及奇数之和。
[root@localhost ~]# vim q.sh
#!/bin/bash
declare i sum=0 sum1
if [[ $1 =~ [^[:digit:]] ]] ;then
echo "$1should be an integer"
exit 5
fi
if [ "$1" -eq 1 ] ; then
echo "奇数和为1"
exit 5
fi
if [ "$1" -eq 2 ] ; then
echo "偶数和为2"
exit 5
fi

for i in $(seq 1 2 $1) ; do
sum=$[sum+i]

done
echo "奇数和为$sum"
for i in $(seq 2  2 $1 ) ; do
sum1=$[sum1+i]

done
echo "偶数的和为$sum1"

结果:[root@localhost ~]# bash q.sh 3
奇数和为4
偶数的和为2
[root@localhost ~]# bash q.sh 5
奇数和为9
偶数的和为6

        例:      
8.利用RANDOM变量随机生成十个数字,显示出这十个数字,并显示出其中的最大值和最小值。

[root@localhost ~]# vim 000

#!/bin/bash

touch ytc
for i in $( seq 1 10 ) ; do
echo $RANDOM &>> ytc
done
sort -n ytc| head -1
sort -n ytc| tail -1
rm -r ytc

结果:
[root@localhost ~]# bash 000
7119
28639
[root@localhost ~]# bash 000
412
31953

 例:给脚本传递一个数字作为行总数,分别打印由*组成的最小锐角朝上和朝下的等腰三角形以及菱形。
  角朝下 

[root@localhost ~]# vim 9
#!/bin/bash

if [ $# -ne 1 ] ; then
echo " usage: $(basename $0) integer"
exit 5
fi

if [[ $1 =~ [^[:digit:]] ]] ; then
echo " usage: $(basename $0) integer"
exit 5
fi

linenum=$1
for i in $(seq $linenum) ; do
for j in $(seq $[i-1 ]); do
echo -n " "
done
for k in $(seq $[2(linenum-i)+1 ]) ; do
echo -n "
"
done
echo
done

结果:[root@localhost ~]# bash 9 8





*******
 *****
  ***
   *

角朝上:
[root@localhost ~]# vim 10
#!/bin/bash

if [ $# -ne 1 ] ; then
echo "Usage: $(basename $0) INTEGER"
exit 5
fi

if [[ $1 =~ [^[:digit:]] ]] ; then
echo "Usage: $(basename $0) INTEGER"
exit 6
fi

linenum=$1
for i in $( seq $linenum) ; do
for j in $( seq $[linenum-i]) ; do
echo -n " "
done
for k in $(seq $[2i-1]) ; do
echo -n "
"
done
echo
done

结果:
[root@localhost ~]# bash 10 8
*


 *****
*******




菱形:
[root@localhost ~]# vim 11
#!/bin/bash

if [ $# -ne 2 ] ; then
echo "Usage: $(basename $0) INTEGER"
exit 5
fi

if [[ $1 =~ [^[:digit:]] ]] ; then
echo "Usage: $(basename $0) INTEGER"
exit 6
fi

linenum=$1
for i in $( seq $linenum) ; do
for j in $( seq $[linenum-i]) ; do
echo -n " "
done
for k in $(seq $[2i-1]) ; do
echo -n "
"
done
echo
done

linenum=$2

for i in $(seq $linenum) ; do
for j in $(seq $[i ]); do
echo -n " "
done
for k in $(seq $[2(linenum-i)+1 ]) ; do
echo -n "
"
done
echo
done

结果:
[root@localhost ~]# bash 11 8 7
*


 *****
*******







*******
 *****
  ***
   *

例:分别打印顺序和旋转的九九乘法表。
1)顺序的九九乘法表是正常的九九乘法表;
[root@localhost ~]# vim 8
#!/bin/bash

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

   2)旋转的九九乘法表是第一行是1×1=1 1×2=2 1×3=3 1×4=4 ... 1×9=9; 第二行是2×2=4 2×3=6 2×4=8 ... 2×9=18; ... 第九行是9×9=81;
 [root@localhost ~]# vim 7
  #!/bin/bash

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

  结果:
  [root@localhost ~]# bash 7

1×1=1 1×2=2 1×3=3 1×4=4 1×5=5 1×6=6 1×7=7 1×8=8 1×9=9
2×2=4 2×3=6 2×4=8 2×5=10 2×6=12 2×7=14 2×8=16 2×9=18
3×3=9 3×4=12 3×5=15 3×6=18 3×7=21 3×8=24 3×9=27
4×4=16 4×5=20 4×6=24 4×7=28 4×8=32 4×9=36
5×5=25 5×6=30 5×7=35 5×8=40 5×9=45
6×6=36 6×7=42 6×8=48 6×9=54
7×7=49 7×8=56 7×9=63
8×8=64 8×9=72
9×9=81

     总结:
         1.进入循环的条件:LIST中还有未被取尽的元素;
         2.退出循环的条件:LIST中的元素被取尽;
         3.for循环几乎不会出现死循环;
         4.在执行循环的过程中,需要将整个LIST载入内存,因此对于大列表来说,可能会消耗较多的内存及CPU资源;

编程思想:
将人类的自然语言转换成程序的代码语言的方式;

DevOPS

不会开发的运维是没有出路的。

流程图

如果 a大于3, 那么
创建一个用户
fi

转载于:https://blog.51cto.com/yuantianchi/2046509

你可能感兴趣的:(开发工具,shell,运维)