shell中$(( ))、$( )與${ }的區別
阿新 • • 發佈:2019-02-12
在bash中,$( )
與` `(反引號)都是用來作命令替換的。
命令替換與變數替換差不多,都是用來重組命令列的,先完成引號裡的命令列,然後將其結果替換出來,再重組成新的命令列。
例子:
$ echo today is $(date "+%Y-%m-%d")
today is 2014-07-01
$( )與 ` `
在操作上,這兩者都是達到相應的效果,但是建議使用$()
,理由如下:
- ““很容易與”搞混亂,尤其對初學者來說。
- 在多層次的複合替換中,
```必須要額外的跳脫處理(反斜線),而
$()`比較直觀。 - 最後,
$()
的弊端是,並不是所有的類unix系統都支援這種方式,但反引號是肯定支援的。
例子:
# 將cmd1執行結果作為cmd2引數,再將cmd2結果作為cmd3的引數
cmd3 $(cmd2 $(cmd1))
# 如果是用反引號,直接引用是不行的,還需要作跳脫處理
cmd3 `cmd2 \`cmd1\``
${ }
變數替換
一般情況下,$var
與${var}
是沒有區別的,但是用${ }
會比較精確的界定變數名稱的範圍
例子:
$ A=B
$ echo ${A}B
BB
取路徑、檔名、字尾
先賦值一個變數為一個路徑,如下:
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} |
若 |
賦值為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 $a
6
$ ((a--));echo $a
5
$ ((a<b));echo $?
0
使用(( ))作整數測試時,不要跟[ ]的整數測試搞混亂了。