淺析Bash中的 {花括號}
讓我們繼續我們的 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 會將這兩個符號之間的輸出內容組合到一起。
另外,各個命令之間要用分號 ;
分隔,否則命令無法正常運行。