淺析 Bash 中的 {花括弧}
在前面的 Bash 基礎系列文章中,我們或多或少地使用了一些還沒有講到的符號。在之前文章的很多例子中,我們都使用到了括弧,但並沒有重點講解關於括弧的內容。
這個系列接下來的文章中,我們會研究括弧們的用法:如何使用這些括弧?將它們放在不同的位置會有什麼不同的效果?除了圓括弧、方括弧、花括弧以外,我們還會接觸另外的將一些內容「包裹」起來的符號,例如單引號、雙引號和反引號。
在這周,我們先來看看花括弧 {}
。
構造序列
花括弧在之前的《點的含義》這篇文章中已經出現過了,當時我們只對點號 .
的用法作了介紹。但在構建一個序列的過程中,同樣不可以缺少花括弧。
我們使用
echo {0..10}
來順序輸出 0 到 10 這 11 個數。使用
echo {10..0}
可以將這 11 個數倒序輸出。更進一步,可以使用
echo {10..0..2}
來跳過其中的奇數。
而
echo {z..a..2}
則從倒序輸出字母表,並跳過其中的第奇數個字母。
以此類推。
還可以將兩個序列進行組合:
echo {a..z}{a..z}
這個命令會將從 aa 到 zz 的所有雙字母組合依次輸出。
這是很有用的。在 Bash 中,定義一個數組的方法是在圓括弧 ()
中放置各個元素並使用空格隔開,就像這樣:
month=("Jan" "Feb" "Mar" "Apr" "May" "Jun" "Jul" "Aug" "Sep" "Oct" "Nov" "Dec")
如果需要獲取數組中的元素,就要使用方括弧 []
並在其中填入元素的索引:
$ echo ${month[3]} # 數組索引從 0 開始,因此 [3] 對應第 4 個元素
Apr
先不要過分關注這裡用到的三種括弧,我們等下會講到。
注意,像上面這樣,我們可以定義這樣一個數組:
letter_combos=({a..z}{a..z})
其中 letter_combos
變數指向的數組依次包含了從 aa 到 zz 的所有雙字母組合。
因此,還可以這樣定義一個數組:
dec2bin=({0..1}{0..1}{0..1}{0..1}{0..1}{0..1}{0..1}{0..1})
在這裡,dec2bin
變數指向的數組按照升序依次包含了所有 8 位的二進位數,也就是 00000000、00000001、00000010,……,11111111。這個數組可以作為一個十進位數到 8 位二進位數的轉換器。例如將十進位數 25 轉換為二進位數,可以這樣執行:
$ echo ${dec2bin[25]}
00011001
對於進位轉換,確實還有更好的方法,但這不失為一個有趣的方法。
參數展開
再看回前面的
echo ${month[3]}
在這裡,花括弧的作用就不是構造序列了,而是用於 參數展開 。顧名思義,參數展開就是將花括弧中的變數展開為這個變數實際的內容。
我們繼續使用上面的 month
數組來舉例:
month=("Jan" "Feb" "Mar" "Apr" "May" "Jun" "Jul" "Aug" "Sep" "Oct" "Nov" "Dec")
注意,Bash 中的數組索引從 0 開始,因此 3 代表第 4 個元素 "Apr"
。因此 echo ${month[3]}
在經過參數展開之後,相當於 echo "Apr"
。
像上面這樣將一個數組展開成它所有的元素,只是參數展開的其中一種用法。另外,還可以通過參數展開的方式讀取一個字元串變數,並對其進行處理。
例如對於以下這個變數:
a="Too longgg"
如果執行:
echo ${a%gg}
可以輸出 「too long」,也就是去掉了最後的兩個 g。
在這裡,
${...}
告訴 shell 展開花括弧里的內容a
就是需要操作的變數%
告訴 shell 需要在展開字元串之後從字元串的末尾去掉某些內容gg
是被去掉的內容
這個特性在轉換文件格式的時候會比較有用,我來舉個例子:
ImageMagick 是一套可以用於操作圖像文件的命令行工具,它有一個 convert
命令。這個 convert
命令的作用是可以為某個格式的圖像文件製作一個另一格式的副本。
下面這個命令就是使用 convert
為 JPEG 格式圖像 image.jpg
製作一個 PNG 格式的圖像副本 image.png
:
convert image.jpg image.png
在很多 Linux 發行版中都預裝了 ImageMagick,如果沒有預裝,一般可以在發行版對應的軟體管理器中找到。
繼續來看,在對變數進行展開之後,就可以批量執行相類似的操作了:
i=image.jpg
convert $i ${i%jpg}png
這實際上是將變數 i
末尾的 "jpg"
去掉,然後加上 "png"
,最終將整個命令拼接成 convert image.jpg image.png
。
如果你覺得並不怎麼樣,可以想像一下有成百上千個圖像文件需要進行這個操作,而僅僅運行:
for i in *.jpg; do convert $i ${i%jpg}png; done
就瞬間完成任務了。
如果需要去掉字元串開頭的部分,就要將上面的 %
改成 #
了:
$ a="Hello World!"
$ echo Goodbye${a#Hello}
Goodbye World!
參數展開還有很多用法,但一般在寫腳本的時候才會需要用到。在這個系列以後的文章中就繼續提到。
合併輸出
最後介紹一個花括弧的用法,這個用法很簡單,就是可以將多個命令的輸出合併在一起。首先看下面這個命令:
echo "I found all these PNGs:"; find . -iname "*.png"; echo "Within this bunch of files:"; ls > PNGs.txt
以分號分隔開的幾條命令都會執行,但只有最後的 ls
命令的結果輸出會被重定向到 PNGs.txt
文件中。如果將這幾條命令用花括弧包裹起來,就像這樣:
{ echo "I found all these PNGs:"; find . -iname "*.png"; echo "Within this bunch of files:"; ls; } > PNGs.txt
執行完畢後,可以看到 PNGs.txt
文件中會包含兩次 echo
的內容、find
命令查找到的 PNG 文件以及最後的 ls
命令結果。
需要注意的是,花括弧與命令之間需要有空格隔開。因為這裡的花括弧 {
和 }
是作為 shell 中的保留字,shell 會將這兩個符號之間的輸出內容組合到一起。
另外,各個命令之間要用分號 ;
分隔,否則命令無法正常運行。
下期預告
在後續的文章中,我會介紹其它「包裹」類符號的用法,敬請關注。
via: https://www.linux.com/blog/learn/2019/2/all-about-curly-braces-bash
作者:Paul Brown 選題:lujun9972 譯者:HankChow 校對:wxy
本文轉載來自 Linux 中國: https://github.com/Linux-CN/archive