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.
#!/bin/bash#this shell test mysql user exist system.if id mysql ;then echo "mysql exist."fi
雙分支if語句
if [ 條件 ];then
語句...
else
語句...
fi
這種雙分支的語句,等條件滿足的時候就會運行then後面的語句,條件不滿足的時候就運行else後面的語句。
eg1:如果指定的使用者存,先說明其已經存在,並顯示其ID號和SHELL;否則,就添加使用者,並顯示其ID號;
#!/bin/bashUsername=mysqlif id $Username &>/dev/null;then Id=`grep "^$Username\>" /etc/passwd | cut -d: -f3` Shell=`grep "^$Username\>" /etc/passwd | cut -d: -f7` echo "$Username exist,ID is $Id,shell is $Shell."else useradd $Username Id=`grep "^$Username\>" /etc/passwd | cut -d: -f3` echo "$Username ID is $Id."fi
eg2:判斷使用者是否存在,存在則顯示ID和SHELL,不存在就添加使用者,並顯示其ID。
分析:這題與上一個列子不一樣的地方是這個使用者沒有指定,既然沒有指定,就需要自己手動去shell一個參數,而指令碼怎麼去引用參數。這裡就用到了bash變數中的位置變數
位置變數:$1,$2.....$9($1就是指令碼傳遞的第一個參數...)
$0:指令碼自身名稱
$@:所有位置參數的列表
$*:所有位置參數
$#:位置參數的個數
#!/bin/bashif id $1 &>/dev/null;then Id=`grep "^$1\>" /etc/passwd | cut -d: -f3` Shell=`grep "^$1\>" /etc/passwd | cut -d: -f7` echo "$1 exist,ID is $Id,shell is $Shell."else useradd $1 Id=`grep "^$1\>" /etc/passwd | cut -d: -f3` echo "$1 ID is $Id."fi
運行指令碼 bash eg2.sh mysql eg.sh是指令碼的名稱,mysql是傳遞給指令碼的第一個參數)
eg3:通過參數傳遞一系列使用者名稱給指令碼,讓指令碼添加這些使用者;但要先判斷使用者是否存在,不存在而後再添加;添加完成後,顯示一共添加了幾個使用者;當然,不能包括因為事先存在而沒有添加的;
分析:這個指令碼不明確的是不知道要傳遞幾個參數,所以用位置變數有點不符合要求,所以引用了$@這個特殊的變數,然後這些使用者,你得一個一個的去取這些使用者,然後進行判斷是否存在,用迴圈控制了如何一個一個的取這些使用者,然後使用者不能重複的一個一個的取,所以使用者得取一個T掉$@列表中的使用者,用了shift命令,shift命令就是輪流的T除使用者。
#!/bin/bashcount=0for user in $@;do if id $1 &>/dev/null;then echo "user $1 exsit." shift else useradd $1 count=$[$count+1] echo "add $1." shift fidoneecho "total users:$count"
運行指令碼
650) this.width=650;" src="http://www.bkjia.com/uploads/allimg/131227/1I239D41-0.png" title="映像 17.png" />
嵌套if語句(最常見的嵌套)
if [ 條件1 ];then
if [ 條件2 ];then
語句...
else
語句.....
fi
else
語句....
fi
這種嵌套的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 ]
#!/bin/bashif id $1 &> /dev/null;then Id=`grep "^$1\>" /etc/passwd | cut -d: -f3` if [ $Id -ge 500 ];then echo "$1 is common user." elif [ $Id -lt 500 -a $Id -ge 1 ];then echo "$1 is system user." else echo "$1 is admin user." fielse echo "$1 Not exist."fi
指令碼執行結果
650) this.width=650;" src="http://www.bkjia.com/uploads/allimg/131227/1I2391X6-1.png" title="映像 18.png" />
eg5:判定使用者的shell是否為登入shell;
分析:首先判斷使用者的shell是否存在,redhat中有一個使用者的shell就為空白,如果shell存在則判斷shell是否為bash,如果是則表示為登入使用者,如果不是則表示為不能登入使用者。在redhat 5.x的版本上。
字串比較
str1 = str2 檢查str1與str2是否相同
str1 != str2 檢查str1與str2是否不同
str1 < str2 檢查str1是否小於str2
str1 > str2 檢查str1是否大於str2
-n str1 檢查str1的長度是否大於0
-z str1 檢查str1的長度是否為0
=~:判斷左邊的字串是否能夠被右邊的模式所匹配,通常用於[[]]; [[ "$opt1" =~ pattern ]],
一般錨定行首和行尾
#!/bin/bashShell=`grep "^$1:" /etc/passwd | cut -d: -f7`if [ -z $Shell ];then echo "$1 user no shell"else if [ "$Shell" == "/bin/bash" ]; then echo "$1 user shell is login shell." else echo "$1 user shell is nologin shell." fifi
運行結果
650) this.width=650;" src="http://www.bkjia.com/uploads/allimg/131227/1I23a408-2.png" title="映像 19.png" />
eg6:判斷當前主機的CPU生產商,其資訊在/proc/cpuinfo檔案中vendor id一行中。如果其生產商為GenuineIntel,就顯示其為Intel公司;否則,就顯示其為AMD公司;
#!/bin/bashVendor=`grep "vendor_id" /proc/cpuinfo | uniq | cut -d: -f2`if [[ "$Vendor" =~ [[:space:]]*GenuineIntel$ ]]; then echo "Intel"else echo "AMD"fi
運行結果
[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 檢查是否存在並且預設組是否為目前使用者組
#!/bin/bash#Author:litaotaodir=`basename $0`if [ $1 == "start" ];then touch /var/lock/subsys/$dir echo "Starting script successfully."elif [ $1 == "stop" ];then rm /var/lock/subsys/$dir &> /dev/null echo "Stop script finished."elif [ $1 == "restart" ];then rm /var/lock/subsys/$dir &> /dev/null touch /var/lock/subsys/$dir echo "Restarting script sucessfully."elif [ $1 == "status" ];then if [ -e /var/lock/subsys/$dir ];then echo "script is running." else echo "script is stopped." fielse echo "$dir.sh {statr|stop|restart|status}"fi
運行方式 bash eg6.sh start 各位博友可以自己把shell copy後執行一遍分析結果
本文出自 “技術之路---桀” 部落格,請務必保留此出處http://litaotao.blog.51cto.com/6224470/1253744