天天看點

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,如需轉載請自行聯系原作者