1. 程式人生 > >Shell指令碼IF條件判斷和判斷條件總結

Shell指令碼IF條件判斷和判斷條件總結

1、基本語法: if [ command ]; then 符合該條件執行的語句 fi 2、擴充套件語法: if [ command ];then 符合該條件執行的語句 elif [ command ];then 符合該條件執行的語句 else 符合該條件執行的語句 fi 3、語法說明: bash shell會按順序執行if語句,如果command執行後且它的返回狀態是0,則會執行符合該條件執行的語句,否則後面的命令不執行,跳到下一條命令。 當有多個巢狀時,只有第一個返回0退出狀態的命令會導致符合該條件執行的語句部分被執行,如果所有的語句的執行狀態都不為0,則執行else中語句。 返回狀態:最後一個命令的退出狀態,或者當沒有條件是真的話為0。 注意: 1、[ ]表示條件測試。注意這裡的空格很重要。要注意在'['後面和']'前面都必須要有空格 2、在shell中,then和fi是分開的語句。如果要在同一行裡面輸入,則需要用分號將他們隔開。 3、注意if判斷中對於變數的處理,需要加引號,以免一些不必要的錯誤。沒有加雙引號會在一些含空格等的字串變數判斷的時候產生錯誤。比如[ -n "$var" ]如果var為空會出錯 4、判斷是不支援浮點值的 5、如果只單獨使用>或者<號,系統會認為是輸出或者輸入重定向,雖然結果顯示正確,但是其實是錯誤的,因此要對這些符號進行轉意 6、在預設中,執行if語句中的命令所產生的錯誤資訊仍然出現在指令碼的輸出結果中 7、使用-z或者-n來檢查長度的時候,沒有定義的變數也為0 8、空變數和沒有初始化的變數可能會對shell指令碼測試產生災難性的影響,因此在不確定變數的內容的時候,在測試號前使用-n或者-z測試一下 9、? 變數包含了之前執行命令的退出狀態(最近完成的前臺程序)(可以用於檢測退出狀態) 常用引數: 檔案/目錄判斷: [ -a FILE ] 如果 FILE 存在則為真。 [ -b FILE ] 如果 FILE 存在且是一個塊檔案則返回為真。 [ -c FILE ] 如果 FILE 存在且是一個字元檔案則返回為真。 [ -d FILE ] 如果 FILE 存在且是一個目錄則返回為真。 [ -e FILE ] 如果 指定的檔案或目錄存在時返回為真。 [ -f FILE ] 如果 FILE 存在且是一個普通檔案則返回為真。 [ -g FILE ] 如果 FILE 存在且設定了SGID則返回為真。 [ -h FILE ] 如果 FILE 存在且是一個符號符號連結檔案則返回為真。(該選項在一些老系統上無效) [ -k FILE ] 如果 FILE 存在且已經設定了冒險位則返回為真。 [ -p FILE ] 如果 FILE 存並且是命令管道時返回為真。 [ -r FILE ] 如果 FILE 存在且是可讀的則返回為真。 [ -s FILE ] 如果 FILE 存在且大小非0時為真則返回為真。 [ -u FILE ] 如果 FILE 存在且設定了SUID位時返回為真。 [ -w FILE ] 如果 FILE 存在且是可寫的則返回為真。(一個目錄為了它的內容被訪問必然是可執行的) [ -x FILE ] 如果 FILE 存在且是可執行的則返回為真。 [ -O FILE ] 如果 FILE 存在且屬有效使用者ID則返回為真。 [ -G FILE ] 如果 FILE 存在且預設組為當前組則返回為真。(只檢查系統預設組) [ -L FILE ] 如果 FILE 存在且是一個符號連線則返回為真。 [ -N FILE ] 如果 FILE 存在 and has been mod如果ied since it was last read則返回為真。 [ -S FILE ] 如果 FILE 存在且是一個套接字則返回為真。 [ FILE1 -nt FILE2 ] 如果 FILE1 比 FILE2 新, 或者 FILE1 存在但是 FILE2 不存在則返回為真。 [ FILE1 -ot FILE2 ] 如果 FILE1 比 FILE2 老, 或者 FILE2 存在但是 FILE1 不存在則返回為真。 [ FILE1 -ef FILE2 ] 如果 FILE1 和 FILE2 指向相同的裝置和節點號則返回為真。 字串判斷 [ -z STRING ] 如果STRING的長度為零則返回為真,即空是真 [ -n STRING ] 如果STRING的長度非零則返回為真,即非空是真 [ STRING1 ]  如果字串不為空則返回為真,與-n類似 [ STRING1 == STRING2 ] 如果兩個字串相同則返回為真 [ STRING1 != STRING2 ] 如果字串不相同則返回為真 [ STRING1 < STRING2 ] 如果 “STRING1”字典排序在“STRING2”前面則返回為真。 [ STRING1 > STRING2 ] 如果 “STRING1”字典排序在“STRING2”後面則返回為真。 數值判斷 [ INT1 -eq INT2 ] INT1和INT2兩數相等返回為真 ,= [ INT1 -ne INT2 ] INT1和INT2兩數不等返回為真 ,<> [ INT1 -gt INT2 ] INT1大於INT2返回為真 ,> [ INT1 -ge INT2 ] INT1大於等於INT2返回為真,>= [ INT1 -lt INT2 ] INT1小於INT2返回為真 ,< [ INT1 -le INT2 ] INT1小於等於INT2返回為真,<= 邏輯判斷 [ ! EXPR ] 邏輯非,如果 EXPR 是false則返回為真。 [ EXPR1 -a EXPR2 ] 邏輯與,如果 EXPR1 and EXPR2 全真則返回為真。 [ EXPR1 -o EXPR2 ] 邏輯或,如果 EXPR1 或者 EXPR2 為真則返回為真。 [ ] || [ ] 用OR來合併兩個條件 [ ] && [ ] 用AND來合併兩個條件 其他判斷 [ -t FD ] 如果檔案描述符 FD (預設值為1)開啟且指向一個終端則返回為真 [ -o optionname ] 如果shell選項optionname開啟則返回為真 IF高階特性: 雙圓括號(( )):表示數學表示式 在判斷命令中只允許在比較中進行簡單的算術操作,而雙圓括號提供更多的數學符號,而且在雙圓括號裡面的'>','<'號不需要轉意。 雙方括號[[ ]]:表示高階字串處理函式 雙方括號中判斷命令使用標準的字串比較,還可以使用匹配模式,從而定義與字串相匹配的正則表示式。 雙括號的作用: 在shell中,[ $a != 1 || $b = 2 ]是不允許出,要用[ $a != 1 ] || [ $b = 2 ],而雙括號就可以解決這個問題的,[[ $a != 1 || $b = 2 ]]。又比如這個[ "$a" -lt "$b" ],也可以改成雙括號的形式(("$a" < "$b")) 例項 1:判斷目錄$doiido是否存在,若不存在,則新建一個 if [ ! -d "$doiido"]; then   mkdir "$doiido" fi 2:判斷普通檔案$doiido是否存,若不存在,則新建一個 if [ ! -f "$doiido" ]; then   touch "$doiido" fi 3:判斷$doiido是否存在並且是否具有可執行許可權 if [ ! -x "$doiido"]; then   mkdir "$doiido" chmod +x "$doiido" fi 4:是判斷變數$doiido是否有值 if [ ! -n "$doiido" ]; then   echo "$doiido is empty"   exit 0 fi 5:兩個變數判斷是否相等 if [ "$var1" = "$var2" ]; then   echo '$var1 eq $var2' else   echo '$var1 not eq $var2' fi 6:測試退出狀態: if [ $? -eq 0 ];then echo 'That is ok' fi 7:數值的比較: if [ "$num" -gt "150" ] echo "$num is biger than 150" fi 8:a>b且a<c (( a > b )) && (( a < c )) [[ $a > $b ]] && [[ $a < $c ]] [ $a -gt $b -a $a -lt $c ] 9:a>b或a<c

?

1

2

3

(( a > b )) || (( a < c ))

[[ $a > $b ]] || [[ $a < $c ]]

[ $a -gt $b -o $a -lt $c ]

10:檢測執行指令碼的使用者

?

1

2

3

4

if [ "$(whoami)" != 'root' ]; then

echo "You have no permission to run $0 as non-root user."

exit

1;

fi

上面的語句也可以使用以下的精簡語句 [ "$(whoami)" != 'root' ] && ( echo "You have no permission to run $0 as non-root user."; exit 1 ) 11:正則表示式

?

1

2

3

4

doiido="hero"

if [[ "$doiido" == h* ]];then

echo "hello,hero"

fi

============其他例子============ 1、檢視當前作業系統型別

?

1

2

3

4

5

6

7

8

9

10

11

#!/bin/sh

SYSTEM=`uname -s`

if [ $SYSTEM = "Linux" ] ; then

echo "Linux"

elif [ $SYSTEM = "FreeBSD" ] ; then

echo "FreeBSD"

elif [ $SYSTEM = "Solaris" ] ; then

echo "Solaris"

else

echo "What?"

fi

2、if利用read傳參判斷

?

1

2

3

4

5

6

7

8

9

10

11

12

13

#!/bin/bash

read -p "please input a score:" score

echo -e "your score [$score] is judging by sys now"

if [ "$score" -ge "0" ]&&[ "$score" -lt "60" ];then

echo "sorry,you are lost!"

elif [ "$score" -ge "60" ]&&[ "$score" -lt "85" ];then

echo "just soso!"

elif [ "$score" -le "100" ]&&[ "$score" -ge "85" ];then

echo "good job!"

else

echo "input score is wrong , the range is [0-100]!"

fi

3、判斷檔案是否存在

?

1

2

3

4

5

6

7

8

9

10

#!/bin/sh

today=`date -d yesterday +%y%m%d`

file="apache_$today.tar.gz"

cd /home/chenshuo/shell

if [ -f "$file" ];then

echo "OK"

else

echo "error $file" >error.log

mail -s "fail backup from test" loveyasxn924@126.com <error.log

fi

4、這個指令碼在每個星期天由cron來執行。如果星期的數是偶數,他就提醒你把垃圾箱清理:

?

1

2

3

4

5

6

#!/bin/bash

WEEKOFFSET=$[ $(date +"%V") % 2 ]

if [ $WEEKOFFSET -eq "0" ]; then

echo "Sunday evening, put out the garbage cans." | mail -s "Garbage cans out" your@your_domain.org

fi

5、掛載硬碟指令碼(windows下的ntfs格式硬碟)

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

#! /bin/sh

dir_d=/media/disk_d

dir_e=/media/disk_e

dir_f=/media/disk_f

a=`ls $dir_d | wc -l`

b=`ls $dir_e | wc -l`

c=`ls $dir_f | wc -l`

echo "checking disk_d..."

if [ $a -eq 0 ]; then

echo "disk_d is not exsit,now creating..."

sudo mount -t ntfs /dev/disk/by-label/software /media/disk_d

else

echo "disk_d exits"

fi

echo "checking disk_e..."

if [ $b -eq 0 ]; then

echo "disk_e is not exsit,now creating..."

sudo mount -t ntfs /dev/disk/by-label/elitor /media/disk_e

else

echo "disk_e exits"

fi

echo "checking disk_f..."

if [ $c -eq 0 ]; then

echo "disk_f is not exsit,now creating..."

sudo mount -t ntfs /dev/disk/by-label/work /media/disk_f

else

echo "disk_f exits"

fi