1. 程式人生 > >shell中$(( ))、$( )與${ }的區別

shell中$(( ))、$( )與${ }的區別

在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,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 $a
6
$ ((a--));echo $a
5
$ ((a<b));echo $?
0

使用(( ))作整數測試時,不要跟[ ]的整數測試搞混亂了。

參考