UNIX 技巧: UNIX 高手的另外 10 個(gè)習(xí)慣
讓我們面對(duì)現(xiàn)實(shí)吧:壞習(xí)慣很難改變。但是您已經(jīng)熟悉的習(xí)慣可能更難克服。有時(shí),重新審視某些事情可能讓您遇到“啊哈,我沒想到它能做到這一點(diǎn)!的時(shí)刻。在 Michael Stutz 的優(yōu)秀文章“Unix 高手的 10 個(gè)習(xí)慣的基礎(chǔ)上,本文將提供另外 10 個(gè) UNIX 命令行命令、工具和技術(shù),可以使您成為更高效的 UNIX 命令行高手。
您應(yīng)當(dāng)采納的其他 10 個(gè)好習(xí)慣包括:
使用文件名自動(dòng)完成功能 (file name completion)。
使用歷史擴(kuò)展。
重用以前的參數(shù)。
使用 pushd 和 popd 管理目錄導(dǎo)航。
查找大型文件。
不使用編輯器創(chuàng)建臨時(shí)文件。
使用 curl 命令行實(shí)用工具。
最有效地利用正則表達(dá)式。
確定當(dāng)前用戶。
使用 awk 處理數(shù)據(jù)。
使用文件名完成
如果不需要在命令提示符處鍵入長(zhǎng)的、令人費(fèi)解的文件名,這是不是很棒呢?的確,您不需要這樣做。相反,您可以配置最流行的 UNIX Shell 以使用文件名完成。該功能在各個(gè) Shell 中的工作方式略有不同,因此我將向您展示如何在最流行的 Shell 中使用文件名完成。文件名完成使您可以更快地輸入并避免錯(cuò)誤。懶惰?也許吧。效率更高?當(dāng)然!
我正在運(yùn)行哪種 Shell?
如果您不知道目前使用的是哪一種 Shell,會(huì)怎么樣?雖然這個(gè)訣竅不是另外 10 個(gè)好習(xí)慣的正式組成部分,但它仍然很有用。如清單 1 所示,您可以使用 echo $0 或 ps -p $$ 命令顯示您正在使用的 Shell。對(duì)于我來說,運(yùn)行的是 Bash Shell。
清單 1. 確定您的 Shell
$ echo $0-bash$ ps –p $$PID TTY TIME CMD6344 ttys0000:00.02 –bash
C Shell
C Shell 支持最直接文件名完成功能。設(shè)置 filec 變量可啟用該功能。(您可以使用命令 set filec。)在您開始鍵入文件名后,可以按 Esc 鍵,Shell 將完成文件名,或完成盡可能多的部分。例如,假設(shè)您擁有名為 file1、file2 和 file3 的文件。如果您鍵入 f,然后按 Esc 鍵,將填充 file,而您必須鍵入 1、2 或 3 來完成相應(yīng)的文件名。
Bash
Bash Shell 也提供了文件名完成,但使用 Tab 鍵代替 Esc 鍵。您在 Bash Shell 中不需要設(shè)置任何選項(xiàng)即可啟用文件名完成,該選項(xiàng)是缺省設(shè)置的。Bash 還實(shí)現(xiàn)了其他功能。鍵入文件名的一部分后,按 Tab 鍵,如果有多個(gè)文件滿足您的請(qǐng)求,并且您需要添加文本以選擇其中一個(gè)文件,那么您可以多按 Tab 鍵兩次,以顯示與您目前鍵入的內(nèi)容相匹配的文件的列表。使用之前名為 file1、file2 和 file3 的文件示例,首先鍵入 f。當(dāng)您按一次 Tab 鍵時(shí),Bash 完成 file;再按一次 Tab 鍵時(shí),將展開列表 file1 file2 file3。
Korn Shell
對(duì)于 Korn Shell 用戶,文件名完成取決于 EDITOR 變量的值。如果 EDITOR 設(shè)置為 vi,那么您鍵入部分名稱,然后按 Esc 鍵,后跟反斜杠 () 字符。如果 EDITOR 設(shè)置為 eMacs,那么您鍵入部分名稱,然后按兩次 Esc 鍵以完成文件名。
使用歷史擴(kuò)展
如果您為一系列命令使用相同的文件名,會(huì)發(fā)生什么情況?當(dāng)然,有一種快捷方式可以快速獲得您上次使用的文件名。如清單 2 所示,!$ 命令返回前一個(gè)命令使用的文件名。從文件 this-is-a-long-lunch-menu-file.txt 中搜索單詞 pickles 的出現(xiàn)位置。搜索結(jié)束后,使用 vi 命令來編輯 this-is-a-long-lunch-menu-file.txt 文件,而不需要重新鍵入文件名。您使用感嘆號(hào) (!) 來訪問歷史,然后使用美元符號(hào) ($) 返回前一命令的最后字段。如果您反復(fù)用到長(zhǎng)文件名,那么這是一個(gè)非常好的工具。
清單 2. 使用 !$ 獲得前一個(gè)命令使用的文件名
$ grep pickles this-is-a-long-lunch-menu-file.txtpastrami on rye with pickles and onions$ vi !$
重用以前的參數(shù)
!$ 命令返回某個(gè)命令使用的上一個(gè)文件名參數(shù)。但如果某個(gè)命令使用多個(gè)文件名,而您只希望重用其中一個(gè)文件名,該如何做?!:1 操作符返回某個(gè)命令使用的第一個(gè)文件名。清單 3 中的示例顯示可以如何將此操作符與 !$ 運(yùn)算符組合使用。在第一個(gè)命令中,將一個(gè)文件重新命名為更有意義的名稱,但為了保持原始文件名可用,創(chuàng)建了一個(gè)符號(hào)鏈接。重新命名文件 kxp12.c 以提高可讀性,然后使用 link 命令來創(chuàng)建到原始文件名的符號(hào)鏈接,以防在其他位置使用該文件名。!$ 操作符返回 file_system_access.c 文件名,而 !:1 操作符返回 kxp12.c 文件名,該文件名是上個(gè)命令的第一個(gè)文件名。
清單 3. 組合使用 !$ 和 !:1
$ mv kxp12.c file_system_access.c$ ln –s !$ !:1
使用 pushd 和 popd 管理目錄導(dǎo)航
UNIX 支持各種目錄導(dǎo)航工具。我最喜歡的兩款提高工作效率的工具是 pushd 和 popd。您當(dāng)然了解 cd 命令用于更改您的當(dāng)前目錄。如果您要在多個(gè)目錄中導(dǎo)航,但希望能夠快速返回某個(gè)位置,該如何做?pushd 和 popd 命令創(chuàng)建一個(gè)虛擬目錄堆棧,pushd 命令用來更改您的當(dāng)前目錄并將其存儲(chǔ)在堆棧中,而 popd 命令用來從堆棧的頂部移除目錄并使您返回該位置。您可以使用 dirs 命令來顯示當(dāng)前目錄堆棧,而不會(huì)壓入或彈出新目錄。清單 4 顯示如何使用 pushd 和 popd 命令在目錄樹中快速導(dǎo)航。
清單 4. 使用 pushd 和 popd 在目錄樹中導(dǎo)航
$ pushd .~ ~$ pushd /etc/etc ~ ~$ pushd /var/var /etc ~ ~$ pushd /usr/local/bin/usr/local/bin /var /etc ~ ~$ dirs/usr/local/bin /var /etc ~ ~$ popd/var /etc ~ ~$ popd/etc ~ ~$ popd~ ~$ popd
pushd 和 popd 命令還支持使用參數(shù)處理目錄堆棧。使用 +n 或 -n 參數(shù),其中 n 是一個(gè)數(shù)字,您可以向左或向右移動(dòng)堆棧,如清單 5 所示。
清單 5. 旋轉(zhuǎn)目錄堆棧
$ dirs/usr/local/bin /var /etc ~ ~$ pushd +1/var /etc ~ ~ /usr/local/bin$ pushd -1~ /usr/local/bin /var /etc ~
查找大型文件
是否需要找出您的所有空閑磁盤空間被什么占用了?您可以使用以下幾個(gè)工具來管理您的存儲(chǔ)設(shè)備。如清單 6 所示,df 命令為您顯示每個(gè)可用卷上已使用的塊的總數(shù),以及空閑空間的百分比。
清單 6. 確定卷的使用情況
$ dfFilesystem512-blocks Used Available Capacity Mounted on/dev/disk0s2 311909984 267275264 4412272086%/devfs224 224 0 100%/devfdesc 2 2 0 100%/devmap -hosts 0 0 0 100%/netmap auto_home 0 0 0 100%/home
是否希望查找大型文件?使用 find 命令時(shí)附帶 -size 參數(shù)。清單 7 顯示了如何使用 find 命令來查找大于 10MB 的文件。請(qǐng)注意,-size 參數(shù)以 KB 為單位計(jì)量大小。
清單 7. 查找大于 10MB 的所有文件
$ find / -size +10000k –xdev –exec ls –lh {};
不使用編輯器創(chuàng)建臨時(shí)文件
以下是一個(gè)簡(jiǎn)單示例:您需要快速創(chuàng)建一個(gè)簡(jiǎn)單臨時(shí)文件,但不希望啟動(dòng)您的編輯器。使用帶有 > 文件重定向操作符的 cat 命令。如清單 8 所示,使用不帶文件名的 cat 命令只回顯向標(biāo)準(zhǔn)輸入鍵入的任何內(nèi)容;> 重定向?qū)⒃撦斎氩东@到指定的文件中。請(qǐng)注意,您在結(jié)束鍵入時(shí)必須提供文件結(jié)束字符,通常為 Ctrl-D。
清單 8. 快速創(chuàng)建臨時(shí)文件
$ cat > my_temp_file.txtThis is my temp file text^D$ cat my_temp_file.txtThis is my temp file text
需要執(zhí)行相同操作,但是附加到現(xiàn)有文件而不是創(chuàng)建新文件。如清單 9 所示,改用 >> 操作符。>> 文件重定向操作符向現(xiàn)有文件附加內(nèi)容。
清單 9.快速向文件附加內(nèi)容
$ cat >> my_temp_file.txtMore text^D$ cat my_temp_file.txtThis is my temp file textMore text
使用 curl 命令行實(shí)用工具
我是否可以從命令行訪問 Web?你瘋了嗎?沒有,這就是 curl 的用途!curl 命令使您可以使用 HTTP、HTTPS、FTP、FTPS、Gopher、DICT、TELNET、LDAP 或 FILE 協(xié)議從服務(wù)器檢索數(shù)據(jù)。如清單 10 所示,我可以使用 curl 命令從美國國家氣象局了解我所在位置(紐約州布法羅市)的當(dāng)前天氣狀況。當(dāng)與 grep 命令組合使用時(shí),我可以檢索布法羅市的天氣狀況。使用 -s 命令行選項(xiàng)來禁止 curl 處理輸出。
清單 10. 使用 curl 檢索當(dāng)前天氣狀況
$ curl –s http://www.srh.noaa.gov/data/ALY/RWRALY | grep BUFFALOBUFFALOMOSUNNY 43 22 43 NE13 30.10R
如清單 11 所示,您也可以使用 curl 命令來下載 HTTP 托管的文件。使用 -o 參數(shù)來指定保存輸出的位置。
清單 11. 使用 curl 下載 HTTP 承載的文件
$ curl -o archive.tar http://www.somesite.com/archive.tar
這實(shí)際上只是您使用 curl 命令可以完成的操作的提示。您只需在命令提示符處鍵入 man curl 顯示 curl 命令的完整使用信息,就可以開始了解更多內(nèi)容。
最有效地利用正則表達(dá)式
大量 UNIX 命令使用正則表達(dá)式作為參數(shù)。從技術(shù)角度而言,正則表達(dá)式 是表示某種模式的字符串(也就是說,由字母、數(shù)字和符號(hào)組成的字符序列),用于定義零或更長(zhǎng)的字符串。正則表達(dá)式使用元字符(例如,星號(hào) [*] 和問號(hào) [?])來匹配其他字符串的部分或全部?jī)?nèi)容。正則表達(dá)式不一定包含通配符,但通配符可以使正則表達(dá)式在搜索模式和處理文件時(shí)發(fā)揮更大的作用。表 1 顯示了一些基本正則表達(dá)式序列。
表 1. 正則表達(dá)式序列
序列說明脫字符 (^)匹配出現(xiàn)在行首的表達(dá)式,例如 ^A 美元符號(hào) ($)匹配出現(xiàn)在行末的表達(dá)式,例如 A$ 反斜杠 ()取消下一個(gè)字符的特殊含義,例如 ^ 方括號(hào) ([])匹配括起來的任一字符,例如 [aeiou](使用連字符 [-] 表示范圍,例如 [0-9])。[^ ] 匹配除括起來字符以外的任一字符,例如 [^0-9] 句點(diǎn) (.)匹配除行尾之外的任意單個(gè)字符星號(hào) (*)匹配零個(gè)或多個(gè)前驅(qū)字符或表達(dá)式{x,y} 匹配出現(xiàn)過 x 到 y 個(gè)和前面相同的內(nèi)容{x} 精確匹配出現(xiàn)過 x 個(gè)和前面相同的內(nèi)容{x,} 匹配出現(xiàn)過 x 個(gè)或更多和前面相同的內(nèi)容
清單 12 顯示了與 grep 命令一起使用的一些基本正則表達(dá)式。
清單 12. 使用正則表達(dá)式和 grep
$ # Lists your mail$ grep '^From: ' /usr/mail/$USER $ # Any line with at least one letter $ grep '[a-zA-Z]' search-file.txt$ # Anything not a letter or number$ grep '[^a-zA-Z0-9] search-file.txt$ # Find phone numbers in the form 999-9999$ grep '[0-9]{3}-[0-9]{4}' search-file.txt$ # Find lines with exactly one character$ grep '^.$' search-file.txt$ # Find any line that starts with a period "." $ grep '^.' search-file.txt$ # Find lines that start with a "." and 2 lowercase letters$ grep '^.[a-z][a-z]' search-file.txt
有大量書籍專門講述正則表達(dá)式。有關(guān)命令行正則表達(dá)式的深入描述,建議您閱讀 developerWorks 文章“對(duì)話 UNIX,第 9 部分:正則表達(dá)式。
確定當(dāng)前用戶
有時(shí),您可能希望確定某個(gè)特定用戶是否運(yùn)行過您的管理腳本。為找出答案,您可以使用 whoami 命令來返回當(dāng)前用戶的名稱。清單 13 顯示了獨(dú)自運(yùn)行的 whoami 命令;清單 14 顯示了使用 whoami 確保當(dāng)前用戶不是根用戶的 Bash 腳本的摘錄。
清單 13. 從命令行使用 whoami
$ whoamiJohn
清單 14. 在腳本中使用 whoami
if [ $(whoami) = "root" ]then echo "You cannot run this script as root." exit 1fi
使用 awk 處理數(shù)據(jù)
awk 命令似乎始終處在 Perl 的陰影下,但它對(duì)于簡(jiǎn)單、基于命令行的數(shù)據(jù)處理來說是一個(gè)快速、實(shí)用的工具。清單 15 顯示了如何開始使用 awk 命令。若要獲取文件中每行文本的長(zhǎng)度,請(qǐng)使用 length() 函數(shù)。若要查看字符串 ing 是否出現(xiàn)在文件文本中,請(qǐng)使用 index() 函數(shù),該函數(shù)返回 ing 首次出現(xiàn)的位置,這樣您就可以使用它來進(jìn)行進(jìn)一步的字符串處理。若要 tokenize(也就是說,將一行拆分為單詞長(zhǎng)度的片段)某個(gè)字符串,請(qǐng)使用 split() 函數(shù)。
清單 15. 基本 awk 處理
$ cat texttesting the awk command$ awk '{ i = length($0); print i }' text23$ awk '{ i = index($0,ing); print i}' text5$ awk 'BEGIN { i = 1 } { n = split($0,a," "); while (i <= n) {print a[i]; i++;} }' texttestingtheawkcommand
打印文本文件中的指定字段是一項(xiàng)簡(jiǎn)單的 awk 任務(wù)。在清單 16 中,sales 文件包含每個(gè)銷售人員的姓名,后跟每月銷售數(shù)字。您可以使用 awk 命令來快速獲得每個(gè)月的銷售總額。缺省情況下,awk 將每個(gè)以逗號(hào)分隔的值視為不同的字段。您使用 $n 操作符來訪問每個(gè)字段。
清單 16. 使用 awk 對(duì)數(shù)據(jù)進(jìn)行匯總
$cat salesGene,12,23,7Dawn,10,25,15Renee,15,13,18David,8,21,17$ awk -F, '{print $1,$2+$3+$4}' salesGene 42Dawn 50Renee 46David 46
awk 命令可以很復(fù)雜并應(yīng)用于廣泛的情景中。若要更完整地學(xué)習(xí) awk 命令,請(qǐng)從命令 man awk 開始,并參閱參考資料部分提供的資源。
結(jié)束語
成為命令行高手需要進(jìn)行一些實(shí)踐。按照相同的方式處理問題很簡(jiǎn)單,因?yàn)槟呀?jīng)習(xí)慣了。擴(kuò)展您的命令行資源可以顯著提高您的工作效率,并促使您朝著 UNIX 命令行高手的方向前進(jìn)!
