我們上一章介紹了( )與{ }的不同,這次讓我們擴展一下,看看更多的變化:$( )與${ }又是啥玩意兒呢?
在bash shell中,$( )與` ` (反引號)都是用來做命令替換用(command substitution)的。
所謂的命令替換與我們第五章學過的變量替換差不多,都是用來重組命令行:
*完成引號裏的命令行,然後將其結果替換出來,再重組命令行。
例如:
$ echo the last sunday is $(date -d "last sunday" +%Y-%m-%d)
如此便可方便得到上一星期天的日期了...
上例是在linux下,在FreeBSD下應該用下面的:
echo the last Sunday is `date –v Sunday +%Y%m%d`
在操作上,用$( )或` `都無所謂,只是我"個人"比較喜歡用$( ),理由是:
1, ` `很容易與' ' (單引號)搞混亂,尤其對初學者來說。
有時在一些奇怪的字形顯示中,兩種符號是一模一樣的(直豎兩點)。
當然了,有經驗的朋友還是一眼就能分辯兩者。只是,若能更好的避免混亂,又何樂不爲呢?
2,在多層次的複合替換中,` `須要額外的跳脫( /` )處理,而$( )則比較直觀。例如:
這是錯的:
command1 `command2 `command3` `
原本的意圖是要在command2 `command3`先將command3提換出來給command 2處理,然後再將結果傳給command1 `command2 ...`來處理。
然而,真正的結果在命令行中卻是分成了`command2 `與``兩段。
正確的輸入應該如下:
command1 `command2 /`command3/` `
要不然,換成$( )就沒問題了:
command1 $(command2 $(command3))
只要你喜歡,做多少層的替換都沒問題啦~~~ ^_^
不過,$( )並不是沒有弊端的...
首先,` `基本上可用在全部的unix shell中使用,若寫成shell script,其移植性比較高。
而$( )並不見的每一種shell都能使用,我只能跟你說,若你用bash2的話,肯定沒問題...
接下來,再讓我們看${ }吧...它其實就是用來作變量替換用的啦。
一般情況下,$var與${var}並沒有啥不一樣。
但是用${ }會比較精確的界定變量名稱的範圍,比方說:
$ A=B
$ echo $AB
原本是打算先將$A的結果替換出來,然後再補一個B字母於其後,但在命令行上,真正的結果卻是隻會替換變量名稱爲AB的值出來...
若使用${ }就沒問題了:
$ echo ${A}B
BB
不過,假如你只看到${ }只能用來界定變量名稱的話,那你就實在太小看bash了﹗
有興趣的話,你可先參考一下cu本版的精華文章:
http://www.chinaunix.net/forum/viewtopic.php?t=201843
爲了完整起見,我這裏再用一些例子加以說明${ }的一些特異功能:
假設我們定義了一個變量爲:
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