最近發現shell腳本在日常工做中簡直算一把瑞士軍刀,不少場景下用shell腳本能實現經常使用的簡單需求,而以前都沒怎麼學習過shell,就趁機把shell相關的語法和常見用法總結了下,方便之後隨時回顧和查閱!html
其實shell的開始並不難,第一行以下:
#!/bin/bash
叫作shebangpython
shell的註釋以#
開頭(這點相似於python)linux
變量定義:
var_test1="hello"
注意定義和賦值時=
兩邊均不能有空格,且變量名前不能加$
!
變量使用:
echo $var_test1
或
echo ${var_test1}
或:
str1="${var_test1} world"
加上美圓符後的變量會被替換爲其值。注意,單引號中的不會被替換
命令執行結果賦給一個變量:web
lines=`wc -l 1.txt` # 這裏lines就被賦值爲了1.txt的行數 lines="$(wc -l 1.txt)" # 單撇號不能嵌套,可用這種包含在內
格式化變量:
echo 23 | awk {printf("%05d", $0);} # 00023
字符串截取:
${var_test1:1:3}
會獲得"ell"
,即從索引1開始取3個字符正則表達式
shell中的輸出直接用echo便可
echo "hello word"
等同於
echo hello word
shell
echo "What's your name?" read Name # 這裏不須要聲明新變量 echo Hello, $Name!
&&
: 知足短路原則,即第一個命令返回失敗時不會執行第二個命令||
: 知足短路原則,即第一個命令返回成功時不會執行第二個命令;
: 不短路,無論第一個命令是否返回成功,均會執行第二個命令echo "Always executed" || echo "Only executed if first command fails" echo "Always executed" && echo "Only executed if first command does NOT fail" echo "Always executed1" ; echo "Always executed2"
cmd1 | cmd2
: 通道,第一個命令的輸出會直接做爲第二個命令的標準輸入來執行
python test.py < in.txt
: 輸入重定向,將從in.txt讀取輸入而非標準輸入流
echo "hello" > out.txt
: 輸出重定向,將前一個命令的輸出定向到文件out.txt中(建立新文件並寫入)
echo "hello" >> append.txt
: 輸出重定向,將前一個命令的輸出定向到文件append.txt中(追加到文件末尾)
echo "hello" 2> err.txt
: 錯誤流重定向,將前一個命令的錯誤流輸出定向到文件err.txt中(建立新文件並寫入)
echo "hello" 2>> err.txt
: 錯誤流重定向,將前一個命令的錯誤流輸出定向到文件err.txt中(追加到文件末尾)bash
最簡單的判斷邏輯是:app
if [ condition1 ]; then statement1 elif [ condition2 ]; then statement2 else statement3 fi # 或者用test if test condition1 ; then statement1 fi
注意:condition兩邊與[]之間必須至少有一個空格less
shell中判斷條件不一樣於其餘語言中的<,>,==,<=,>=那麼直接,常見的有下列幾種:oop
-eq 等於(==) -ne 不等於(!=) -gt 大於(\>) -lt 小於(\<) -le 小於等於 -ge 大於等於 -z 空串 -n 非空串 == 兩個字符相等 != 兩個字符不等
-a 且 -o 或
[[ ... ]]
[[ hello == hell? ]]
結果爲真[[ ]]
條件判斷結構中,可是若是出如今 [ ]
結構中的話,會報錯。好比能夠直接使用 if [[ $a != 1 && $a <= 5 ]]
, 若是不使用雙括號, 則爲 if [ $a -ne 1] && [ $a -le 5 ]
或者 if [ $a -ne 1 -a $a -le 5 ]
-f 判斷後面是否爲一個文件 -d 判斷後面是否爲一個目錄 -e 判斷後面對應的文件是否存在 -s 判斷文件是否存在且不爲空
case "$Variable" in # 列出須要匹配的字符串 0) echo "There is a zero.";; 1) echo "There is a one.";; *) echo "It is not null.";; esac
# {1..3} == `seq 1 3` for Variable in {1..3} do echo "$Variable" done
或傳統的"for循環",但須要加兩層括號(兩層的小括號內能夠寫C語言中的語句):
for ((a=1; a <= 3; a++)) do echo $a done
在其餘命令的結果上執行for循環:
for Output in $(ls) do cat "$Output" done
while [ condition ] do echo "loop body here..." break done
$# 命令行參數個數 $0 當前腳本名 $n 第n個參數值,n可取1,2,3... $@ 全部命令行參數 $? 上一個命令的返回值
# 打印每行中 ',' 以前內容 cut -d ',' -f 1 file.txt # 將 file.txt 文件全部 'okay' 替換爲 'great', (兼容正則表達式) sed -i 's/okay/great/g' file.txt # shell中不支持浮點數除法運算,可以使用awk實現浮點除法 a=3 b=4 c=`awk 'BEGIN{printf "%.2f",('$a'/'$b')}'` # 單引號內的變量不能被替換,所以須要將變量單獨放在引號外 sort -k 指定比較的列(從1開始) -n 數值比較 -r 倒序 -o filename 輸出到文件(可用此選項輸出到輸入文件) -f 不區分大小寫排序 -c 檢查是否已排序好,若是未排序好則輸出第一個未按序的行 -M 按月份排序 -b 忽略前導空格 -u 獲得不重複的行 多列排序:sort -k1,1 -k3nr,3
命令 | 做用 |
---|---|
${#string} |
$string 的長度 |
${string:position} |
在$string 中, 從位置$position 開始提取子串 |
${string:position:length} |
在$string 中, 從位置$position 開始提取長度爲$length 的子串 |
${string#substring} |
從變量$string 的開頭, 刪除最短匹配$substring 的子串 |
${string##substring} |
從變量$string 的開頭, 刪除最長匹配$substring 的子串 |
${string%substring} |
從變量$string 的結尾, 刪除最短匹配$substring 的子串 |
${string%%substring} |
從變量$string 的結尾, 刪除最長匹配$substring 的子串 |
${string/substring/replacement} |
使用$replacement , 來代替第一個匹配的$substring |
${string//substring/replacement} |
使用$replacement , 代替全部匹配的$substring |
${string/#substring/replacement} |
若是$string 的前綴匹配$substring , 那麼就用$replacement 來代替匹配到的$substring |
${string/%substring/replacement} |
若是$string 的後綴匹配$substring , 那麼就用$replacement 來代替匹配到的$substring |
a="one,two,three,four" # 要將$a分割開,能夠這樣: OLD_IFS="$IFS" IFS="," arr=($a) IFS="$OLD_IFS" for s in ${arr[@]} do echo "$s" done # 後面的能夠直接改成只取${arr[1]}
加粗的部分支持管道接收標準輸入流。更多命令可參考:Linux命令大全
本文簡要介紹了下shell入門的一些用法,若有不對之處,歡迎你們指正。我也是不久前纔開始真正使用shell,這是個強大而又複雜的語言,你們有比較好的shell資料也歡迎留言,共同窗習,謝謝!
轉載請註明出處:Shell常見用法小記