感嘆號(hào):bash 的歷史擴(kuò)展功能
Bash 的歷史擴(kuò)展(History Expansion)又被稱為 Bang(!) 命令,歷史擴(kuò)展是 bash 將歷史命令轉(zhuǎn)換到可執(zhí)行命令的過程。Bash 下的 History 庫提供了一個(gè)與 csh 下歷史擴(kuò)展類似的歷史擴(kuò)展功能。歷史擴(kuò)展中操作歷史命令一般有兩個(gè)部分:
- 首先要從歷史命令中找出相對應(yīng)的命令,被選擇到的命令我們稱作為Event(條目),比如Bang Bang(!!),就是選擇***一條命令;
- 選擇選定行的部分或全部文本以包含到當(dāng)前行中。要操作的條目(Event)Bash將其拆分成了Words(詞),命令中的Words是靠空格來分割的,我們就可以使用修飾符(Modifiers)來調(diào)整Words以符合我們的要求。注意:Words并不是英文單詞,而是一個(gè)字符序列而已。
先來看兩個(gè)命令,你知道第二個(gè)命令是什么意思么?
cat /tmp/cat.cat.txt
!:0 !*:gs/cat./echo.
條目標(biāo)志符(Event Designators)
條目標(biāo)志符是一個(gè)到歷史列表內(nèi)一個(gè)命令行實(shí)體的引用,除非是絕對引用,不然條目的引用是相對歷史列表中當(dāng)前位置的。
條目標(biāo)志符 | 條目標(biāo)志符說明 |
---|---|
! |
開始一個(gè)歷史替換,除非后面緊跟的是空格,制表符,行結(jié)束符,"=","("(當(dāng)使用內(nèi)建命令shopt 開啟了extglob 的shell選項(xiàng))。 |
!n |
重復(fù)歷史中編號(hào)為n的命令——歷史編號(hào)可以參看history 命令. |
!-n |
執(zhí)行之前的第n條命令,執(zhí)行上一條命令可以使用!!或者!-1,執(zhí)行之前第三條命令:!-3,倒推的列表是history 。 |
!! |
執(zhí)行上一條命令,和Ctrl-P,!-1的作用一樣。 |
!string |
執(zhí)行最近的以string字串開頭的命令。這個(gè)命令的意思是重復(fù)以!后字串開頭的***一條命令,比如:!ca將重復(fù)以字符ca開頭的***一條命令,如cat ReadMe ,(假設(shè)最近一條ca開頭是這個(gè)命令,并且ReadMe后緊跟換行符) |
!?string[?] |
在歷史列表中以當(dāng)前位置開始向后查找(往回搜索)包含string字符串的最近一條命令,如果要查找的string字符串后面緊跟換行符,則string后面的這個(gè)問號(hào)可以省略。例如:!?Read?還是會(huì)匹配cat ReadMe 。(同上的環(huán)境),如果后面是換行符如:!?ReadMe,則不用輸入結(jié)尾的[?]。 |
^a^b |
快速替換,把上一條命令中的a替換成b,并執(zhí)行替換后的命令。^a^b^類似。注意:這里只是替換一個(gè)找到的實(shí)例,相當(dāng)于:!!:s/a/b 。 |
^abc |
刪除上一條命令中的abc。 |
!# |
引用目前輸入的所有字串,如:more a !# ;這個(gè)最終的命令是more a more a 。 |
詞標(biāo)志符(Word Designators)
詞標(biāo)志符被用來在條目里面選擇需要的詞。一般用":"分隔條目指示符和詞指示符。當(dāng)詞指示符是以"^","$","*","-","%"開頭時(shí),也可能會(huì)省略":"。詞是從一行的行首開始,***個(gè)詞編號(hào)為0.插入到當(dāng)前行中時(shí),這些詞使用單個(gè)空格分隔。
詞標(biāo)志符 | 詞標(biāo)志符說明 |
---|---|
0 |
第0個(gè)詞,在很多應(yīng)用程序中,這就是命令本身。 |
n |
第n個(gè)詞 |
^ |
***個(gè)參數(shù);也就是***個(gè)詞。 |
$ |
***一個(gè)參數(shù)。 |
% |
最近"?string?"匹配的詞。 |
x-y |
詞的范圍:如果是'0-y'可以簡寫成'-y'. |
* |
除了第0個(gè)以外的所有詞,這個(gè)和'1-$'同義,如果條目中只有一個(gè)詞,使用'*'也不會(huì)返回錯(cuò)誤,僅是返回一個(gè)空字符串而已。 |
x* |
'x-$'的簡寫 |
x- |
和x*類似,都是'x-$'的簡寫,不過需要注意,這個(gè)寫法是忽略***一個(gè)詞的。 |
需要注意的是,在Bash下使用詞指示符的時(shí)候,可以沒有條目指示符,如果沒有使用條目指示符,則會(huì)把前一條命令作為詞指示符的操作條目。
修飾符(Modifiers)
在可選的詞指示符之后,你可以添加下面修飾符中的一個(gè)或多個(gè),每個(gè)修飾符以':'開頭。
修飾符 | 修飾符說明 |
---|---|
h |
去掉路徑名的尾部,只保留頭部。只移除***一個(gè)'/'后面的內(nèi)容,可以理解成是路徑名的父目錄。 |
t |
去掉路徑名部件中除尾部之外的所有內(nèi)容。只保留***一個(gè)'/'后的內(nèi)容。 |
r |
去掉尾部這樣格式".suffix"的一個(gè)結(jié)尾后綴,保留基本名稱。只刪除***一個(gè)點(diǎn)'.'后的內(nèi)容。 |
e |
僅保留后綴。僅保留***一個(gè)點(diǎn)'.'及點(diǎn)后的內(nèi)容。 |
p |
打印新的命令但不執(zhí)行。 |
q |
引用替換的詞,防止進(jìn)一步替換。(譯注,原文:Quote the substituted words, escapin further substitutions.——Mitchell Chu)。這個(gè)引用會(huì)直接對引用的命令加上單引號(hào),防止進(jìn)一步替換。開始這句不知道怎么翻譯。后來Mitchell發(fā)現(xiàn)自己的這個(gè)翻譯并沒有錯(cuò)誤,因?yàn)槲覀円玫脑~可能是個(gè)變量,這時(shí)候如果沒有引號(hào),就會(huì)引起進(jìn)一步的替換,而是用此參數(shù)就能達(dá)到防止這種情況的發(fā)生。 |
x |
這個(gè)和q一樣,是引用替換的詞,但是這個(gè)與q不同的地方在于,q是整體引用,而這個(gè)是會(huì)將替換的詞使用空格,制表符,換行符來分割成一個(gè)個(gè)的詞。 |
s/old/new/ |
把條目行中找到的***個(gè)old位置的內(nèi)容替換成new位置的內(nèi)容,'/'這個(gè)分隔符位置可以使用任何其他字符作為分隔符。如果要在old或new位置使用分隔符,需要使用反斜桿'\'來轉(zhuǎn)義。如果'&'這個(gè)字符出現(xiàn)在new位置,將會(huì)被替換成old位置的內(nèi)容,如果要使用'&'請用'\'轉(zhuǎn)義。***一個(gè)分隔符如果是整行的***一個(gè)字符,則可以省略。 |
& |
重復(fù)上次替換。這個(gè)是引用***一次的s/old/new/內(nèi)容。 |
g |
見下,與a相同 |
a |
使替換在整個(gè)條目中進(jìn)行,和's'一起使用,例如:!!:gs/old/new/ ,或者和'&'一起使用。 |
G |
對條目中的每一個(gè)詞都執(zhí)行一次其后的's'修飾符。這個(gè)方法在Bash 4.1.2下測試并不靠譜。
因此Mitchell在想,是不是僅對參數(shù)執(zhí)行一次,而對命令(第0個(gè)詞)進(jìn)行全局替換。但另外一個(gè)測試,反駁了這個(gè)觀點(diǎn):
但多次測試結(jié)果來看,第零個(gè)詞匯被替換最多兩次,其他只替換一次。具體原因暫時(shí)未知! |
了解了這些,我們來揭曉一下文章開頭的命令的意義:
- 我們首先是選出命令!!(!:0可以寫成!!:0,!*同樣可以寫成!!*)
- 有了命令之后我們選擇第二步,利用0,選擇出詞(!:0選擇出來的是cat)
- 第三步是對詞進(jìn)行操作,這里是!*后面對參數(shù)進(jìn)行了字符替換。
- ***變成完成的命令了: cat /tmp/echo.echo.txt
(注:轉(zhuǎn)載時(shí)對原文有修改。)