命令替換
在bash中,$( )與` `(反引號)都是用來作命令替換的。
命令替換與變數替換差不多,都是用來重組命令列的,先完成引號裡的命令列,然後將其結果替換出來,再重組成新的命令列。
exp 1
$ echo today is $(date "+%Y-%m-%d")today is 2014-07-01
$( )與``
在操作上,這兩者都是達到相應的效果,但是建議使用$( ),理由如下: ``很容易與''搞混亂,尤其對初學者來說。 在多層次的複合替換中,``必須要額外的跳脫處理(反斜線),而$( )比較直觀。 最後,$( )的弊端是,並不是所有的類unix系統都支援這種方式,但反引號是肯定支援的。
exp 2
# 將cmd1執行結果作為cmd2參數,再將cmd2結果作為cmd3的參數cmd3 $(cmd2 $(cmd1))# 如果是用反引號,直接引用是不行的,還需要作跳脫處理cmd3 `cmd2 \`cmd1\``
${ }變數替換
一般情況下,$var與${var}是沒有區別的,但是用${ }會比較精確的界定變數名稱的範圍
$ A=B$ echo ${A}BBB
取路徑、檔案名稱、尾碼
先賦值一個變數為一個路徑,如下:
file=/dir1/dir2/dir3/my.file.txt
命令 |
解釋 |
結果 |
${file#*/} |
拿掉第一條 / 及其左邊的字串 |
dir1/dir2/dir3/my.file.txt |
${file##*/} |
拿掉最後一條 / 及其左邊的字串 |
my.file.txt |
${file#*.} |
拿掉第一個 . 及其左邊的字串 |
file.txt |
${file##*.} |
拿掉最後一個 . 及其左邊的字串 |
txt |
${file%/*} |
拿掉最後一條 / 及其右邊的字串 |
/dir1/dir2/dir3 |
${file%%/*} |
拿掉第一條 / 及其右邊的字串 |
(空值) |
${file%.*} |
拿掉最後一個 . 及其右邊的字串 |
/dir1/dir2/dir3/my.file |
${file%%.*} |
拿掉第一個 . 及其右邊的字串 |
/dir1/dir2/dir3/my
|
記憶方法如下: # 是去掉左邊(在鍵盤上 # 在 $ 之左邊) % 是去掉右邊(在鍵盤上 % 在 $ 之右邊) 單一符號是最小匹配;兩個符號是最大匹配 *是用來匹配不要的字元,也就是想要去掉的那部分 還有指定字元分隔號,與*配合,決定取哪部分
取子串及替換
命令 |
解釋 |
結果 |
${file:0:5} |
提取最左邊的 5 個位元組 |
/dir1 |
${file:5:5} |
提取第 5 個位元組右邊的連續 5 個位元組 |
/dir2 |
${file/dir/path} |
將第一個 dir 提換為 path |
/path1/dir2/dir3/my.file.txt |
${file//dir/path} |
將全部 dir 提換為 path |
/path1/path2/path3/my.file.txt |
${#file} |
擷取變數長度 |
27
|
根據狀態為變數賦值
命令 |
解釋 |
備忘 |
${file-my.file.txt} |
若 $file 沒設定,則使用 my.file.txt 作傳回值 |
空值及非空值不作處理 |
${file:-my.file.txt} |
若 $file 沒有設定或為空白值,則使用 my.file.txt 作傳回值 |
非空值時不作處理 |
${file+my.file.txt} |
若$file 設為空白值或非空值,均使用my.file.txt作傳回值 |
沒設定時不作處理 |
${file:+my.file.txt} |
若 $file 為非空值,則使用 my.file.txt 作傳回值 |
沒設定及空值不作處理 |
${file=txt} |
若 $file 沒設定,則回傳 txt ,並將 $file 賦值為 txt |
空值及非空值不作處理 |
${file:=txt} |
若 $file 沒設定或空值,則回傳 txt ,將 $file 賦值為txt |
非空值時不作處理 |
${file?my.file.txt} |
若 $file 沒設定,則將 my.file.txt 輸出至 STDERR |
空值及非空值不作處理 |
${file:?my.file.txt} |
若 $file沒設定或空值,則將my.file.txt輸出至STDERR |
非空值時不作處理
|
tips:
以上的理解在於, 你一定要分清楚 unset 與 null 及 non-null 這三種賦值狀態. 一般而言, : 與 null 有關, 若不帶 : 的話, null 不受影響, 若帶 : 則連 null 也受影響.
數組
A="a b c def" # 定義字串A=(a b c def) # 定義字元數組
命令 |
解釋 |
結果 |
${A[@]} |
返回數組全部元素 |
a b c def |
${A[*]} |
同上 |
a b c def |
${A[0]} |
返回數組第一個元素 |
a |
${#A[@]} |
返回數組元素總個數 |
4 |
${#A[*]} |
同上 |
4 |
${#A[3]} |
返回第四個元素的長度,即def的長度 |
3 |
A[3]=xyz |
則是將第四個組數重新定義為 xyz |
|
$(( ))與整數運算
bash中整數運算子號
符號 |
功能 |
+ - * / |
分別為加、減、乘、除 |
% |
餘數運算 |
& | ^ ! |
分別為“AND、OR、XOR、NOT”
|
在 $(( )) 中的變數名稱,可於其前面加 $ 符號來替換,也可以不用。
$ a=5;b=7;c=2$ echo $((a+b*c))19$ echo $(($a+$b*$c))19
進位轉換
$(( ))可以將其他進位轉成十進位數顯示出來。用法如下:
echo $((N#xx))
其中,N為進位,xx為該進位下某個數值,命令執行後可以得到該進位數轉成十進位後的值。
$ echo $((2#110)) # 二進位轉十進位6$ echo $((16#2a)) # 十六進位轉十進位42$ echo $((8#11)) # 八進位轉十進位9
(( ))重定義變數值
$ a=5;b=7$ ((a++));echo $a6$ ((a--));echo $a5$ ((a<b));echo $?0
使用(( ))作整數測試時,不要跟[ ]的整數測試搞混亂了。