本文參考連結:http://www.runoob.com/linux/linux-shell.html
基本說明
Shell指令碼(shell script)是一種為shell編寫的指令碼程式。其中shell常見的種類包括:
- Bourne Shell:對應的命令(/usr/bin/sh或/bin/sh)
- Bourne Again Shell:對應的命令(/bin/bash)
- C Shell:對應的命令(/usr/bin/csh)
- K Shell:對應的命令(/usr/bin/ksh)
- Shell for Root:對應的命令(/sbin/sh)
- ……
Shell指令碼檔案命名沒有特殊要求,跟Linux的要求一樣就行,至於檔案一般以.sh作為字尾以區分其他檔案型別。不加字尾或使用其他字尾也行,只要裡面的文字內容符合Shell指令碼格式要求就行。
Shell指令碼格式要求:第一行開頭必須以#!開頭,這告訴系統其後的路徑(即上面說的不同型別的指令碼環境(解析器)的路徑)所指定的程式即是解釋此指令碼檔案的 Shell 程式。
一個簡單的Shell指令碼如下:
#!/bin/bash //告訴系統這行以下的命令都使用/bin/bash環境執行 echo "Hello World !" //輸出字串
執行Shell指令碼的方法:
1、指令碼檔名作為解析器的引數
root@454009d432a4:/workdir# bash test.sh //使用bash命令執行test.sh指令碼 hello world! //指令碼輸出結果
這種方式執行的指令碼,不需要在第一行指定直譯器資訊,寫了也沒用。
2、作為可執行程式
將檔案新增x可執行許可權,直接執行檔案。
這種指令碼執行方式有一點必須注意,就是Linux只能執行格式為unix格式的指令碼,如果指令碼是在windows下編輯傳到Linux系統下執行的話檔案的格式為doc格式。
要解決這種問題只要將文字格式轉換為unix即可。具體方法為使用vi編輯器開啟檔案,在命令模式中使用命令:set ff=unix修改後儲存退出或者直接使用dos2unix 命令執行 dos2unix test.sh 即可:
#!/bin/bash echo "hello world!"
~
~
:set ff=unix
具體實行命令如下:
root@454009d432a4:/workdir# cd /workdir //進入檔案所在目錄
root@454009d432a4:/workdir# chmod +x ./test.sh
root@454009d432a4:/workdir# ./test.sh bash: ./test.sh: /bin/bash^M: bad interpreter: No such file or directory
注意,一定要寫成 ./test.sh,而不是 test.sh,執行其它二進位制的程式也一樣,直接寫 test.sh,linux 系統會去 PATH 裡尋找有沒有叫 test.sh 的,而只有 /bin, /sbin, /usr/bin,/usr/sbin 等在 PATH 裡,你的當前目錄通常不在 PATH 裡,所以寫成 test.sh 是會找不到命令的,要用 ./test.sh 告訴系統說,就在當前目錄找。
變數使用
定義變數時,變數名不加美元符號($,PHP語言中變數需要)
- 命名只能使用英文字母,數字和下劃線,首個字元不能以數字開頭
- 中間不能有空格,可以使用下劃線”_”
- 不能使用標點符號
- 不能使用bash裡的關鍵字(可用help命令檢視保留關鍵字)
#!/bin/bash echo `----------------------------------------------------` # 變數使用 name="walter" echo "hello ${name}!" # 標識變數為只讀 # readonly name # 標識為readonly的變數不能修改包括unset # 刪除變數 unset name name="666" echo "hello ${name}" echo `----------------------------------------------------` # 字串使用 # Shell字串單雙引號 # 單引號內的字元都是原樣輸出,不能變數注入。 # 單引號內可以出現任意雙引號,原樣輸出。 # 而且單引號內不能存在單獨的單引號,如果是一對單引號則會作為字串拼接的功能出現 # 單引號內不支援單引號轉義輸出 str=`I am ${name} "long time no see" ` echo $str # 雙引號內的字元可以使用${}進行變數注入。 # 雙引號內可以出現任意單引號,原樣輸出。 # 而且雙引號內不能存在單獨的雙引號,如果是一對雙引號則會作為字串拼接的功能出現 # 雙引號內支援雙引號的轉義輸出 str="I`m ${name} "long time no see" " echo $str # 獲取字串長度 echo ${#str} # 擷取字串從索引1開始擷取一直擷取6個字元 echo ${str:1:6} # 反引號的使用:將反引號(` `)內的字串當作shell命令來執行,返回值是命令的執行的結果,起到的是一個命令的替換作用 echo `expr index "$str" l6` echo `----------------------------------------------------` # 陣列使用 # bash只支援以為陣列,不限定陣列大小 # 陣列索引由0開始 # 陣列定義用括號表示,陣列中元素用空格分割 name_list=(walter check ford lucy) # 也可以這樣 name_list=( walter check ford lucy ) name_list[0]=walter name_list[1]=check name_list[2]=ford name_list[3]=lucy echo ${name_list[0]} echo ${name_list[1]} echo ${name_list[2]} echo ${name_list[3]} # 預設返回第一個陣列值 echo $name_list # [0]會作為一個獨立的字串輸出,跟前面的$name_list沒關係 echo $name_list[0] # 使用@或*符號可以獲取陣列中的所有元素 echo ${name_list[*]} echo ${name_list[@]} # 取得陣列元素的個數 echo ${#name_list[@]} # 或者 echo ${#name_list[*]} # 取得陣列單個元素的長度 echo ${#name_list[1]}
呼叫指令碼的時候還可以向指令碼里面傳遞引數,指令碼內獲取引數的格式為:$n。n代表一個數字1為第一個引數,以此類推
root@454009d432a4:/workdir# ./test.sh param1 param2 //呼叫指令碼傳入兩個參
#!/bin/bash echo $1 //第一個參 echo $2 //第二個參
另外還有幾個特殊字元用了處理引數供以後參考:
- $# 傳遞到指令碼的個數
- $* 以一個單字串顯示所有向指令碼傳遞的引數。如”$*”用「”」括起來的情況、以”$1 $2 … $n”的形式輸出所有引數
- $$ 指令碼執行的當前程式ID號
- $! 後臺執行的最後一個程式的ID號
- $@ 與$*相同,但是使用時加引號,並在引號中返回每個引數。如”$@”用「”」括起來的情況、以”$1″ “$2” … “$n” 的形式輸出所有引數。
- $- 顯示Shell使用的當前選項,與set命令功能相同
- $? 顯示最後命令的退出狀態。0表示沒有錯誤,其他任何值表明有錯誤
運算
原生的bash不支援直接數學運算,需要藉助上面說的反引號(` `)通過bash命令expr等進行運算
#!/bin/bash a=10 b=20
#注意expr命令後面的運算表示式中各元素間必須用空格分割開,作為條件表示式時同理
#乘法運算中*號必須通過號進行轉義
#在 MAC 中 shell 的 expr 語法是:$((表示式)),此處表示式中的 "*" 不需要轉義符號 "" result=`expr $a + $b` result=`expr $a - $b` result=`expr $a * $b` result=`expr $a / $b` result=`expr $a % $b`
#expr後面可以接條件表示式但是返回結果的時候與php類似是以1、0表示true、false。
echo `expr $a == $b`
條件表示式
條件表示式中關係運算子只支援數字或數字型別的字串。
關係運算子如下:
- -eq 是否相等
- -ne 是否不相等
- -gt 左邊大於右邊
- -lt 左邊小於右邊
- -ge 左邊大於等於右邊
- -le 左邊小於等於右邊
布林運算子:
- ! 非運算
- -o 或運算
- -a 與運算
邏輯運算子:
- && 邏輯的AND
- || 邏輯的OR
字串運算子:
- = 兩個字串是否相等
- != 兩個字串是否不相等
- -z 字串長度是否為0 為0返回true
- -n 字串長度是否為0 不為0返回true
- str 字串是否為空 不為空返回true
檔案測試運算子,用於檢測unix檔案各種屬性:
- -b 檢測檔案是否是塊裝置檔案 是則返回true
- -c 檢測檔案是否字元裝置檔案 是則返回true
- -d 檢測檔案是否目錄 是則返回true
- -f 檢測檔案是否普通檔案(既不是目錄也不是裝置檔案) 是則返回true
- -g 檢測檔案是否設定了SGID位 是則返回true
- -k 檢測檔案是否設定了粘著位(Sticky Bit) 是則返回true
- -p 檢測檔案是否有管道 是則返回true
- -u 檢測檔案是否設定了SUID位 是則返回true
- -r 檢測檔案是否刻度 是則返回true
- -w 檢測檔案是否可寫 是則返回true
- -x 檢測檔案是否可執行 是則返回true
- -s 檢測檔案是否為空(檔案大小是否大於0) 不為空則返回true
- -e 檢測檔案(包括目錄是否存在) 是則返回true
#!/bin/bash
a=10 b=20 a=$b echo `expr $a == $b`
#以下兩種條件判斷語句的格式都可以,不知道其他版本是不是也這樣支援
#注意在條件判斷語句中表示式必須放於[]中括號中,並且表示式前後與中括號之間要有空格分割
##############if 與 then 在不同行中##################
if [ $a != $b ]
then
#statements
echo `no`
fi
###############if 與 then在同一行中以逗號分割#################
if [ $a == $b ]; then
#statements
echo `yes`
fi
if [ $a = $b ]; then
#statements
echo `yes`
fi
################################
if [ $a -eq $b ]; then
#statements
echo `equal`
else
echo `no equal`
fi
################################
if [ $a -ne $b ]
then
echo `no equal`
else
echo `equal`
fi
################################
file=`/workdir/check.log`
if [ -e $file ]; then
#statements
echo `file exist`
else
echo `file not exist`
fi