shell字串比較、判斷是否為數字
二元比較操作符,比較變數或者比較數字.注意數字與字串的區別.
◆ 整數比較
-eq 等於,如:if [ "$a" -eq "$b" ]
-ne 不等於,如:if [ "$a" -ne "$b" ]
-gt 大於,如:if [ "$a" -gt "$b" ]
-ge 大於等於,如:if [ "$a" -ge "$b" ]
-lt 小於,如:if [ "$a" -lt "$b" ]
-le 小於等於,如:if [ "$a" -le "$b" ]
< 小於(需要雙括弧),如:(("$a" < "$b"))
<= 小於等於(需要雙括弧),如:(("$a" <= "$b"))
> 大於(需要雙括弧),如:(("$a" > "$b"))
>= 大於等於(需要雙括弧),如:(("$a" >= "$b"))
◆ 字串比較
= 等於,如:if [ "$a" = "$b" ]
== 等於,如:if [ "$a" == "$b" ],與=等價
注意:==的功能在[[]]和[]中的行為是不同的,如下:
1 [[ $a == z* ]] # 如果$a以"z"開頭(模式比對)那麼將為true
2 [[ $a == "z*" ]] # 如果$a等於z*(字元匹配),那麼結果為true
3
4 [ $a == z* ] # File globbing 和word splitting將會發生
5 [ "$a" == "z*" ] # 如果$a等於z*(字元匹配),那麼結果為true
一點解釋,關於File globbing是一種關於檔案的速記法,比如"*.c"就是,再如~也是.
但是file globbing並不是嚴格的Regex,雖然絕大多數情況下結構比較像.
!= 不等於,如:if [ "$a" != "$b" ]
這個操作符將在[[]]結構中使用模式比對.
< 小於,在ASCII字母順序下.如:
if [[ "$a" < "$b" ]]
if [ "$a" \< "$b" ]
注意:在[]結構中"<"需要被轉義.
> 大於,在ASCII字母順序下.如:
if [[ "$a" > "$b" ]]
if [ "$a" \> "$b" ]
注意:在[]結構中">"需要被轉義.
具體參考Example 26-11來查看這個操作符應用的例子.
-z 字串為"null".就是長度為0.
-n 字串不為"null"
注意:
使用-n在[]結構中測試必須要用""把變數引起來.使用一個未被""的字串來使用! -z
或者就是未用""引用的字串本身,放到[]結構中。雖然一般情況下可
以工作,但這是不安全的.習慣於使用""來測試字串是一種好習慣.
awk '{print $2}' class.txt | grep '^[0-9.]' > res
◆ SHELL下的數字比較及計算
比較:
● 方法一: if [ ${A} -lt ${B} ]; then ...
這是最基本的比較方法,使用lt(小於),gt(大於),le(小於等於),ge(大於等於),優點:還沒發現;缺點:只能比較整數,使用lt,gt等不直觀
● 方法二: if ((${A} < ${B})) then ...
這是CShell風格比較,優點:不用使用lt,gt等難記的字串;缺點:還是只能比較整數
● 方法三: if (echo ${A} ${B} | awk '!($1>$2){exit 1}') then ...
這是使用awk比較,優點:可以比較小數;缺點:運算式太複雜,難記
● 方法四: if (echo ${A} - ${B} | bc -q | grep -q "^-"); then ...
這是使用bc計算比較,優點:可以比較小數;缺點:運算式更複雜,難記
計算:
● 方法一:typeset C=$(expr ${A} + ${B});
SHELL中的基本工具,優點:方便檢測變數是否為數字;缺點:只能計算整數,且只能計算加減法,不能計算乘除法
● 方法二:let "C=${A}+${B}"; 或 let "C=A+B"
內嵌命令計算,優點:能計算乘除法及位元運算等;缺點:只能計算整數
● 方法三:typeset C=$((A+B))
CShell風格的計算,優點:能計算乘除法及位元運算等,簡介,編寫方便;缺點:不能計算小數
● 方法四:typeset C=${echo ${A} ${B} | awk '{print $1+$2}')
使用awk計算,優點:能計算小數,可以實現多種計算方式,計算靈活;缺點:運算式太複雜
● 方法五:typeset C=${echo ${A} + ${B} | bc -q)
使用awk計算,優點:能計算小數,計算方式比awk還多,計算靈活;缺點:運算式太複雜,小數點後面的位元必須使用scale=N來設定,否則可能會將結果截斷為整數
◆ 特殊字元
符號 使用
; 一般情況我們輸出完一個命令需要按一個斷行符號,如果你想在一行執行多個命令,中間可以用;號分割 cd /home ; ls
* 表示任一字元(正則)
? 任一個字元
[abc] 清單項目之一
[^abc] 對於列表取非 也可以使用範圍 [a-z] [0-9] [A-Z](所有字元和數字)
{} 迴圈列表時用 touch_{1,2,3}時就會建立touch_1,touch_2,touch_3迴圈出這三個檔案,也會用 echo ${ab}c
~ home目錄cd ~ (普通通話進入的是/home目錄下使用者自己的家目錄)
$ 提取變數值
`` $() 命令替換touch `date +%F_\`date +%T\`` touch $(date +%F_$(date +%T))
$[] 整數計算 echo $[2+3] - * / % 浮點數用 echo "scale=3; 10/3" | bc -l (bc用於計算的)
\ 轉義後面的字串 echo \\ 輸出\ 轉義特殊字元,為防止被SHELL解釋bash中的特殊字元
"" '' 帶空格串 將空格視為串的一部分 echo "abc xyz" echo 'abc xyz'
`` 命令替換 取命令的執行結果
$() 同上,但它彌補了``的嵌套缺陷
@ 無特殊含義
# 注釋(一般編程都需要加註釋,讓其他團隊隊員對自己寫的程式功能瞭解)
$ 變數取值
$() 命令替換
${} 變數名的範圍
% 殺後台經常jobs號,模數運算(大家對模數應該並不陌生)
^ 取非 和 !雷同
& 用進程幕後處理, &&用於邏輯與
* 匹配任一字元串;計算乘法
() 子進程執行
- 減號,區間,cd - 回到上層目錄,殺掉當前jobs
_ (底線)無特殊含義
+ 加號; 殺掉當前jobs(進程)
= 賦值
| 管道,|| 邏輯或
\ 轉義 當一些特殊符號如$是一個變數需要轉義才不被bash解析
{} 命令列表 {ls;cd /;}
[] 字元萬用字元,[]也是用於測試命令
: 空命令 真值
; 命令結束符
"" 軟引 '' 硬引
< 輸入重新導向
> 輸出重新導向
>& 合并2和1輸出
, 枚舉分隔字元
. 目前的目錄
/ 目錄分隔字元
? 單個字元
斷行符號 命令執行