天天看点

shell脚本编程之选择控制结构

   shell脚本编程之选择控制结构

      程序开发语言,分为两种,一种是解释型语言,一种是编译型语言。解释型语言是解释语句,并且能根据流程控制机制让语句按需执行,解释一条语句就返回语句的结果,这种语言需要一个解释器,而linux中的bash就是这样的一个解释器,常见的解释型语言有perl,python,ruby,bash。编译型语言,首先需要编译源程序,并且能够将源程序转换为二进制格式,而后让其执行,这种语言需要一个编辑器,如linux下的gcc就是一个编辑器,编译型语言有C,C++,C#等。无论是解释型语言还是编译型语言,都有语言控制结构,没有控制结构的程序,不能算做一个好的程序。

语言控制结构分为三种:

  顺序执行:默认法则,逐条执行各语句

  选择执行:条件判断,只有部分是符合条件的,只执行符合条件的部分

  循环执行:将同一段代码反复的执行n次

这篇博客写关于选择控制结构。

     linux的选择控制结构有两种一种是 if-then,一种是case--esac。

1、if-then

bash条件测试:

[ expression ]

` expression `

test expression

bash命令

if-then有三种结构形式:

单分支if语句

  if [ 条件 ];then

语句...

    fi      

这种单分支的语句,当条件满足的时候,就会运行then后面的语句,不满足就直接退出判断语句

eg:系统中是否存在mysql用户,存在就显示mysql exist.

1

2

3

4

5

<code>#!/bin/bash</code>

<code>#this shell test mysql user exist system.</code>

<code>if</code> <code>id</code> <code>mysql ;</code><code>then</code>

<code>   </code><code>echo</code> <code>"mysql exist."</code>

<code>fi</code>

双分支if语句

if [ 条件 ];then

            语句...

    else

    fi

这种双分支的语句,等条件满足的时候就会运行then后面的语句,条件不满足的时候就运行else后面的语句。

eg1:如果指定的用户存,先说明其已经存在,并显示其ID号和SHELL;否则,就添加用户,并显示其ID号;

6

7

8

9

10

11

<code>Username=mysql</code>

<code>if</code>  <code>id</code> <code>$Username  &amp;&gt;</code><code>/dev/null</code><code>;</code><code>then</code>

<code>   </code><code>Id=`</code><code>grep</code> <code>"^$Username\&gt;"</code> <code>/etc/passwd</code> <code>| </code><code>cut</code> <code>-d: -f3`</code>

<code>   </code><code>Shell=`</code><code>grep</code> <code>"^$Username\&gt;"</code> <code>/etc/passwd</code> <code>| </code><code>cut</code> <code>-d: -f7`</code>

<code>   </code><code>echo</code> <code>"$Username exist,ID is $Id,shell is $Shell."</code>

<code>else</code>

<code>    </code><code>useradd</code> <code>$Username</code>

<code>    </code><code>Id=`</code><code>grep</code> <code>"^$Username\&gt;"</code> <code>/etc/passwd</code> <code>| </code><code>cut</code> <code>-d: -f3`</code>

<code>    </code><code>echo</code> <code>"$Username ID is $Id."</code>

eg2:判断用户是否存在,存在则显示ID和SHELL,不存在就添加用户,并显示其ID。

分析:这题与上一个列子不一样的地方是这个用户没有指定,既然没有指定,就需要自己手动去shell一个参数,而脚本怎么去引用参数。这里就用到了bash变量中的位置变量

  位置变量:$1,$2.....$9($1就是脚本传递的第一个参数...)

          $0:脚本自身名称

          $@:所有位置参数的列表

          $*:所有位置参数

          $#:位置参数的个数

<code>if</code>  <code>id</code> <code>$1  &amp;&gt;</code><code>/dev/null</code><code>;</code><code>then</code>

<code>   </code><code>Id=`</code><code>grep</code> <code>"^$1\&gt;"</code> <code>/etc/passwd</code> <code>| </code><code>cut</code> <code>-d: -f3`</code>

<code>   </code><code>Shell=`</code><code>grep</code> <code>"^$1\&gt;"</code> <code>/etc/passwd</code> <code>| </code><code>cut</code> <code>-d: -f7`</code>

<code>   </code><code>echo</code> <code>"$1 exist,ID is $Id,shell is $Shell."</code>

<code>    </code><code>useradd</code> <code>$1</code>

<code>    </code><code>Id=`</code><code>grep</code> <code>"^$1\&gt;"</code> <code>/etc/passwd</code> <code>| </code><code>cut</code> <code>-d: -f3`</code>

<code>    </code><code>echo</code> <code>"$1 ID is $Id."</code>

运行脚本 bash eg2.sh  mysql    (eg.sh是脚本的名称,mysql是传递给脚本的第一个参数)

eg3:通过参数传递一系列用户名给脚本,让脚本添加这些用户;但要先判断用户是否存在,不存在而后再添加;添加完成后,显示一共添加了几个用户;当然,不能包括因为事先存在而没有添加的;

分析:这个脚本不明确的是不知道要传递几个参数,所以用位置变量有点不符合要求,所以引用了$@这个特殊的变量,然后这些用户,你得一个一个的去取这些用户,然后进行判断是否存在,用循环控制了如何一个一个的取这些用户,然后用户不能重复的一个一个的取,所以用户得取一个T掉$@列表中的用户,用了shift命令,shift命令就是轮流的T除用户。

12

13

14

<code>count=0</code>

<code>for</code> <code>user </code><code>in</code> <code>$@;</code><code>do</code>

<code>  </code><code>if</code> <code>id</code> <code>$1 &amp;&gt;</code><code>/dev/null</code><code>;</code><code>then</code>

<code>    </code><code>echo</code> <code>"user $1 exsit."</code>

<code>    </code><code>shift</code>

<code>  </code><code>else</code>

<code>    </code><code>count=$[$count+1]</code>

<code>   </code><code>echo</code> <code>"add $1."</code>

<code>   </code><code>shift</code>

<code>  </code><code>fi</code>

<code>done</code>

<code>echo</code> <code>"total users:$count"</code>

运行脚本

<a target="_blank" href="http://blog.51cto.com/attachment/201307/113958134.png"></a>

嵌套if语句(最常见的嵌套)

if [ 条件1 ];then

            if [ 条件2 ];then

                  语句...

           else

                  语句.....

fi

   else

              语句....

这种嵌套的if语句就相对复杂一点,当条件1满足时,就运行第一个then后面的语句,在条件1满足后,在判断条件2,当条件1和条件2同时满足的时候就会运行第二个then后面的语句,当只有条件1满足,条件2不满足就会运行第一个else后面的语句,如果条件1都不满足,那就直接运行第二个else后面的语句了。

eg4:判断一个用户,如果存在则判断是否是普通用户,还是系统用户,还是admin用户。不存在则显示该用户不存在。

分析:首先判断用户是否存在,存在则比较用户的ID是否为大于等于500,大于等于500则表示为普通用户,是否1-499,是则表示为系统用户,是否为0,是则为admin用户。

本例子还用到了数值的比较,然后还用到了条件的组合

linux test命令支持数值比较、字符串比较、文件比较 详情请man test

数值比较

    num1 -eq num2 检查num1是否等于num2

    num1 -ge num2 检查num1是否>或等于num2

    num1 -gt num2 检查num1是否大于num2

    num1 -le num2 检查num1是否<等于num2

    num1 -lt num2 检查num1是否<num2

    num1 -ne num2 检查num1是否不等于num2

组合条件测试

   -a:与        [ $Uid -ge 1 -a $Uid -le 499 ]

   -o:或        [ $Uid -eq 0 -a $Uid -ge 500 ]

   !:非,单目操作符 [ ! $Uid -eq 0 ]

<code>if</code> <code>id</code> <code>$1 &amp;&gt; </code><code>/dev/null</code><code>;</code><code>then</code>

<code>    </code><code>if</code> <code>[ $Id -</code><code>ge</code> <code>500 ];</code><code>then</code>

<code>         </code><code>echo</code> <code>"$1 is common user."</code>

<code>    </code><code>elif</code> <code>[ $Id -lt 500 -a $Id -</code><code>ge</code> <code>1 ];</code><code>then</code>

<code>         </code><code>echo</code> <code>"$1 is system user."</code>

<code>    </code><code>else</code>

<code>         </code><code>echo</code> <code>"$1 is admin user."</code>

<code>    </code><code>fi</code>

<code>    </code><code>echo</code> <code>"$1 Not exist."</code>

脚本执行结果

<a target="_blank" href="http://blog.51cto.com/attachment/201307/121056810.png"></a>

eg5:判定用户的shell是否为登录shell;

分析:首先判断用户的shell是否存在,redhat中有一个用户的shell就为空,如果shell存在则判断shell是否为bash,如果是则表示为登录用户,如果不是则表示为不能登录用户。在redhat 5.x的版本上。

字符串比较

  str1  = str2 检查str1与str2是否相同

  str1 != str2 检查str1与str2是否不同

  str1 &lt;  str2 检查str1是否小于str2

  str1 &gt;  str2 检查str1是否大于str2

  -n   str1    检查str1的长度是否大于0

  -z   str1    检查str1的长度是否为0

  =~:判断左边的字符串是否能够被右边的模式所匹配,通常用于[[]]; [[ "$opt1" =~ pattern ]],  

      一般锚定行首和行尾

<code>Shell=`</code><code>grep</code> <code>"^$1:"</code> <code>/etc/passwd</code> <code>| </code><code>cut</code> <code>-d: -f7`</code>

<code>if</code> <code>[ -z $Shell ];</code><code>then</code>

<code>   </code><code>echo</code> <code>"$1 user no shell"</code>

<code>   </code><code>if</code> <code>[ </code><code>"$Shell"</code> <code>== </code><code>"/bin/bash"</code> <code>]; </code><code>then</code>

<code>      </code><code>echo</code> <code>"$1 user shell is login shell."</code>

<code>   </code><code>else</code>

<code>       </code><code>echo</code> <code>"$1 user shell is nologin shell."</code>

<code>   </code><code>fi</code>

运行结果

<a target="_blank" href="http://blog.51cto.com/attachment/201307/122900285.png"></a>

eg6:判断当前主机的CPU生产商,其信息在/proc/cpuinfo文件中vendor id一行中。如果其生产商为GenuineIntel,就显示其为Intel公司;否则,就显示其为AMD公司;

<code>Vendor=`</code><code>grep</code> <code>"vendor_id"</code> <code>/proc/cpuinfo</code>  <code>| </code><code>uniq</code> <code>| </code><code>cut</code> <code>-d: -f2`</code>

<code>if</code> <code>[[ </code><code>"$Vendor"</code> <code>=~ [[:space:]]*GenuineIntel$ ]]; </code><code>then</code>

<code>  </code><code>echo</code> <code>"Intel"</code>

<code>  </code><code>echo</code> <code>"AMD"</code>

[root@Redhat5 test]# bash eg6.sh

Intel

eg7:

写一个脚本:可以接受一个参数,其使用形式如下:

script.sh {start|stop|restart|status}

如果参数为start,创建空文件/var/lock/subsys/script,并显示“Starting script successfully.”;

如果参数为stop,则删除文件/var/lock/subsys/script,并显示“Stop script finished.”;

如果参数为restart,则删除文件/var/lock/subsys/script后重新创建,并显示“Restarting script successfully.”;

如果参数为status,那么:

如果/var/lock/subsys/script文件存在,则显示为“script is running.”

否则,则显示为“script is stopped.”

其它任何参数:则显示“script.sh {start|stop|restart|status}”

文件比较

          -b file            检查文件是否存在且是一个块特殊文件

          -c file            检查文件是否存在且是一个字符文件

          -d file            检查file是否存在并且是一个目录

          -e file            检查文件是否存在

          -f file            检查文件是否存在并且是一个文件

          -h file            检查文件存在且为一个符合链接

          -r file            检查文件是否存在并且可读

          -s file            检查文件是否存在且不为空

  -w file            检查文件是否存在且可写

          -x file            检查文件是否存在且可执行

          -O file            检查文件是否存在且别当前用户拥有

  -G file            检查是否存在并且默认组是否为当前用户组

15

16

17

18

19

20

21

22

<code>#Author:litaotao</code>

<code>dir</code><code>=`</code><code>basename</code> <code>$0`</code>

<code>if</code> <code>[ $1 == </code><code>"start"</code> <code>];</code><code>then</code>

<code>   </code><code>touch</code> <code>/var/lock/subsys/</code><code>$</code><code>dir</code>

<code>   </code><code>echo</code> <code>"Starting script successfully."</code>

<code>elif</code> <code>[ $1 == </code><code>"stop"</code> <code>];</code><code>then</code>

<code>   </code><code>rm</code> <code>/var/lock/subsys/</code><code>$</code><code>dir</code> <code>&amp;&gt; </code><code>/dev/null</code>

<code>   </code><code>echo</code> <code>"Stop script finished."</code>

<code>elif</code> <code>[ $1 == </code><code>"restart"</code> <code>];</code><code>then</code>

<code>  </code><code>rm</code> <code>/var/lock/subsys/</code><code>$</code><code>dir</code> <code>&amp;&gt; </code><code>/dev/null</code>

<code>  </code><code>touch</code> <code>/var/lock/subsys/</code><code>$</code><code>dir</code>

<code>  </code><code>echo</code> <code>"Restarting script sucessfully."</code>

<code>elif</code> <code>[ $1 == </code><code>"status"</code> <code>];</code><code>then</code>

<code>    </code><code>if</code> <code>[ -e </code><code>/var/lock/subsys/</code><code>$</code><code>dir</code> <code>];</code><code>then</code>

<code>       </code><code>echo</code> <code>"script is running."</code>

<code>       </code><code>echo</code> <code>"script is stopped."</code>

<code>   </code><code>echo</code> <code>"$dir.sh {statr|stop|restart|status}"</code>

运行方式   bash eg6.sh start    各位博友可以自己把shell copy后执行一遍分析结果

本文转自 jie783213507 51CTO博客,原文链接:http://blog.51cto.com/litaotao/1253744,如需转载请自行联系原作者