命令替換
在bash中,$( )
與` `
(反引號)都是用來作命令替換的。
命令替換與變量替換差不多,都是用來重組命令行的,先完成引號里的命令行,然后將其結果替換出來,再重組成新的命令行。
exp 1
[root@localhost ~]# echo today is $(date "+%Y-%m-%d") today is 2017-11-07 [root@localhost ~]# echo today is `date "+%Y-%m-%d"` today is 2017-11-07
$( )與``
在操作上,這兩者都是達到相應的效果,但是建議使用$( ),理由如下:
``很容易與''搞混亂,尤其對初學者來說,而$( )比較直觀。
最后,$( )的弊端是,並不是所有的類unix系統都支持這種方式,但反引號是肯定支持的。
exp 2
[root@localhost ~]# echo Linux `echo Shell `echo today is `date "+%Y-%m-%d"``` Linux Shellecho today is 2017-11-07 #過多使用``會有問題 [root@localhost ~]# echo Linux `echo Shell $(echo today is $(date "+%Y-%m-%d"))` Linux Shell today is 2017-11-07 ``和$()混合使用 [root@localhost ~]# echo Linux $(echo Shell $(echo today is $(date "+%Y-%m-%d"))) Linux Shell today is 2017-11-07 #多個$()同時使用也不會有問題
${ }變量替換
一般情況下,$var與${var}是沒有區別的,但是用${ }會比較精確的界定變量名稱的范圍
exp 1
[root@localhost ~]# A=Linux
[root@localhost ~]# echo $AB #表示變量AB [root@localhost ~]# echo ${A}B #表示變量A后連接着B LinuxB
取路徑、文件名、后綴
先賦值一個變量為一個路徑,如下:
file=/dir1/dir2/dir3/my.file.txt 命令 解釋 結果 ${file#*/} 拿掉第一條 / 及其左邊的字符串 dir1/dir2/dir3/my.file.txt [root@localhost ~]# echo ${file#*/} dir1/dir2/dir3/my.file.txt ${file##*/} 拿掉最后一條 / 及其左邊的字符串 my.file.txt [root@localhost ~]# echo ${file##*/} my.file.txt ${file#*.} 拿掉第一個 . 及其左邊的字符串 file.txt [root@localhost ~]# echo ${file#*.} file.txt ${file##*.} 拿掉最后一個 . 及其左邊的字符串 txt [root@localhost ~]# echo ${file##*.} txt ${file%/*} 拿掉最后一條 / 及其右邊的字符串 /dir1/dir2/dir3 [root@localhost ~]# echo ${file%/*} /dir1/dir2/dir3 ${file%%/*} 拿掉第一條 / 及其右邊的字符串 (空值) [root@localhost ~]# echo ${file%%/*} (空值) ${file%.*} 拿掉最后一個 . 及其右邊的字符串 /dir1/dir2/dir3/my.file [root@localhost ~]# echo ${file%.*} /dir1/dir2/dir3/my.file ${file%%.*} 拿掉第一個 . 及其右邊的字符串 /dir1/dir2/dir3/my [root@localhost ~]# echo ${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]=xzy | 則是將第四個組數重新定義為 xyz |
$(( ))與整數運算
bash中整數運算符號
符號 | 功能 |
---|---|
+ - * / | 分別為加、減、乘、除 |
% | 余數運算 |
& | ^ ! | 分別為“AND、OR、XOR、NOT” |
在 $(( )) 中的變量名稱,可於其前面加 $ 符號來替換,也可以不用。
[root@localhost ~]# echo $((2*3)) 6 [root@localhost ~]# a=5;b=7;c=2 [root@localhost ~]# echo $((a+b*c)) 19 [root@localhost ~]# echo $(($a+$b*$c)) 19
進制轉換
$(( ))可以將其他進制轉成十進制數顯示出來。用法如下:echo $((N#xx))
其中,N為進制,xx為該進制下某個數值,命令執行后可以得到該進制數轉成十進制后的值。
[root@localhost ~]# echo $((2#110)) 6 [root@localhost ~]# echo $((16#2a)) 42 [root@localhost ~]# echo $((8#11)) 9
(())重定義變量值
[root@localhost ~]# a=5;b=7 [root@localhost ~]# ((a++)) [root@localhost ~]# echo $a 6 [root@localhost ~]# ((a--));echo $a 5 [root@localhost ~]# ((a<b));echo $? 0 [root@localhost ~]# ((a>b));echo $? 1