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