清單 1 演示了最多見的 UNIX 壞習慣之一:一次定義一個目錄樹。 html
清單 1. 壞習慣 1 的示例:單獨定義每一個目錄樹 正則表達式
~ $ mkdir tmp shell
~ $ cd tmp bash
~/tmp $ mkdir a 服務器
~/tmp $ cd a curl
~/tmp/a $ mkdir b 編輯器
~/tmp/a $ cd b 函數
~/tmp/a/b/ $ mkdir c 工具
~/tmp/a/b/ $ cd c 測試
~/tmp/a/b/c $
使用 mkdir 的 -p 選項並在單個命令中建立全部父目錄及其子目錄要容易得多。可是即便對於知道此選項的管理員,他們在命令行上建立子目錄時也仍然束縛於逐步建立每級子目錄。花時間有意識地養成這個好習慣是值得的.
清單 2. 好習慣 1 的示例:使用一個命令來定義目錄樹
~ $ mkdir -p tmp/a/b/c
您可使用此選項來建立整個複雜的目錄樹(在腳本中使用是很是理想的),而不僅是建立簡單的層次結構。
清單 3. 好習慣 1 的另外一個示例:使用一個命令來定義複雜的目錄樹
~ $ mkdir -p project/{lib/ext,bin,src,doc/{html,info,pdf},demo/stat/a}
過去,單獨定義目錄的惟一藉口是您的 mkdir 實現不支持此選項,可是在大多數系統上再也不是這樣了。IBM、AIX®、mkdir、GNU mkdir 和其餘遵照單一 UNIX 規範 (Single UNIX Specification) 的系統如今都具備此選項。
對於仍然缺少該功能的少數系統,您可使用 mkdirhier 腳本(請參見參考資料),此腳本是執行相同功能的 mkdir 的包裝:
~ $ mkdirhier project/{lib/ext,bin,src,doc/{html,info,pdf},demo/stat/a}
另外一個不良的使用模式是將 .tar 存檔文件移動到某個目錄,由於該目錄剛好是您但願在其中提取 .tar 文件的目錄。其實您根本不須要這樣作。您能夠爲所欲爲地將任何 .tar 存檔文件解壓縮到任何目錄——這就是 -C 選項的用途。在解壓縮某個存檔文件時,使用 -C 選項來指定要在其中解壓縮該文件的目錄:
清單 4. 好習慣 2 的示例:使用選項 -C 來解壓縮 .tar 存檔文件
~ $ tar xvf -C tmp/a/b/c newarc.tar.gz
相對於將存檔文件移動到您但願在其中解壓縮它的位置,切換到該目錄,而後才解壓縮它,養成使用 -C 的習慣則更加可取——當存檔文件位於其餘某個位置時尤爲如此。
您可能已經知道,在大多數 Shell 中,您能夠在單個命令行上經過在命令之間放置一個分號 (;) 來組合命令。該分號是 Shell 控制操做符,雖然它對於在單個命令行上將離散的命令串聯起來頗有用,但它並不適用於全部狀況。例如,假設您使用分號來組合兩個命令,其中第二個命令的正確執行徹底依賴於第一個命令的成功完成。若是第一個命令未按您預期的那樣退出,第二個命令仍然會運行——結果會致使失敗。相反,應該使用更適當的控制操做符(本文將描述其中的部分操做符)。只要您的 Shell 支持它們,就值得養成使用它們的習慣。
使用 && 控制操做符來組合兩個命令,以便僅當 第一個命令返回零退出狀態時才運行第二個命令。換句話說,若是第一個命令運行成功,則第二個命令將運行。若是第一個命令失敗,則第二個命令根本就不運行。例如:
清單 5. 好習慣 3 的示例:將命令與控制操做符組合使用
~ $ cd tmp/a/b/c && tar xvf ~/archive.tar
在此例中,存檔的內容將提取到 ~/tmp/a/b/c 目錄中,除非該目錄不存在。若是該目錄不存在,則 tar 命令不會運行,所以不會提取任何內容。
相似地,|| 控制操做符分隔兩個命令,而且僅當第一個命令返回非零退出狀態時才運行第二個命令。換句話說,若是第一個命令成功,則第二個命令不會運行。若是第一個命令失敗,則第二個命令纔會 運行。在測試某個給定目錄是否存在時,一般使用此操做符,若是該目錄不存在,則建立它:
清單 6. 好習慣 3 的另外一個示例:將命令與控制操做符組合使用
~ $ cd tmp/a/b/c || mkdir -p tmp/a/b/c
您還能夠組合使用本部分中描述的控制操做符。每一個操做符都影響最後的命令運行:
清單 7. 好習慣 3 的組合示例:將命令與控制操做符組合使用
~ $ cd tmp/a/b/c || mkdir -p tmp/a/b/c && tar xvf -C tmp/a/b/c ~/archive.tar
始終要謹慎使用 Shell 擴展和變量名稱。通常最好將變量調用包括在雙引號中,除非您有不這樣作的足夠理由。相似地,若是您直接在字母數字文本後面使用變量名稱,則還要確保將該變量名稱包括在方括號 ([]) 中,以使其與周圍的文本區分開來。不然,Shell 將把尾隨文本解釋爲變量名稱的一部分——而且極可能返回一個空值。清單 8 提供了變量的各類引用和非引用及其影響的示例。
清單 8. 好習慣 4 的示例:引用(和非引用)變量
~ $ ls tmp/
a b
~ $ VAR="tmp/*"
~ $ echo $VAR
tmp/a tmp/b
~ $ echo "$VAR"
tmp/*
~ $ echo $VARa
~ $ echo "$VARa"
~ $ echo "${VAR}a"
tmp/*a
~ $ echo ${VAR}a
tmp/a
~ $
您或許看到過使用反斜槓 (\) 來將較長的行延續到下一行的代碼示例,而且您知道大多數 Shell 都將您經過反斜槓聯接的後續行上鍵入的內容視爲單個長行。然而,您可能沒有在命令行中像一般那樣利用此功能。若是您的終端沒法正確處理多行迴繞,或者您的命令行比一般小(例如在提示符下有長路經的時候),反斜槓就特別有用。反斜槓對於瞭解鍵入的長輸入行的含義也很是有用,如如下示例所示:
清單 9. 好習慣 5 的示例:將反斜槓用於長輸入
~ $ cd tmp/a/b/c || \
> mkdir -p tmp/a/b/c && \
> tar xvf -C tmp/a/b/c ~/archive.tar
或者,也可使用如下配置:
清單 10. 好習慣 5 的替代示例:將反斜槓用於長輸入
~ $ cd tmp/a/b/c \
> || \
> mkdir -p tmp/a/b/c \
> && \
> tar xvf -C tmp/a/b/c ~/archive.tar
然而,當您將輸入行劃分到多行上時,Shell 始終將其視爲單個連續的行,由於它老是刪除全部反斜槓和額外的空格。
注意:在大多數 Shell 中,當您按向上箭頭鍵時,整個多行輸入將重繪到單個長輸入行上。
大多數 Shell 都具備在列表中對命令分組的方法,以便您能將它們的合計輸出向下傳遞到某個管道,或者將其任何部分或所有流重定向到相同的地方。您通常能夠經過在某個 Subshell 中運行一個命令列表或經過在當前 Shell 中運行一個命令列表來實現此目的。
使用括號將命令列表包括在單個組中。這樣作將在一個新的 Subshell 中運行命令,並容許您重定向或收集整組命令的輸出,如如下示例所示:
清單 11. 好習慣 6 的示例:在 Subshell 中運行命令列表
~ $ ( cd tmp/a/b/c/ || mkdir -p tmp/a/b/c && \
> VAR=$PWD; cd ~; tar xvf -C $VAR archive.tar ) \
> | mailx admin -S "Archive contents"
在此示例中,該存檔的內容將提取到 tmp/a/b/c/ 目錄中,同時將分組命令的輸出(包括所提取文件的列表)經過郵件發送到地址 admin。
當您在命令列表中從新定義環境變量,而且您不但願將那些定義應用於當前 Shell 時,使用 Subshell 更可取。
將命令列表用大括號 ({}) 括起來,以在當前 Shell 中運行。確保在括號與實際命令之間包括空格,不然 Shell 可能沒法正確解釋括號。此外,還要確保列表中的最後一個命令以分號結尾,如如下示例所示:
清單 12. 好習慣 6 的另外一個示例:在當前 Shell 中運行命令列表
~ $ { cp ${VAR}a . && chown -R guest.guest a && \
> tar cvf newarchive.tar a; } | mailx admin -S "New archive"
使用 xargs 工具做爲篩選器,以充分利用從 find 命令挑選的輸出。find 運行一般提供與某些條件匹配的文件列表。此列表被傳遞到 xargs 上,後者而後使用該文件列表做爲參數來運行其餘某些有用的命令,如如下示例所示:
清單 13. xargs 工具的經典用法示例
~ $ find some-file-criteria some-file-path | \
> xargs some-great-command-that-needs-filename-arguments
然而,不要將 xargs 僅看做是 find 的輔助工具;它是一個未獲得充分利用的工具之一,當您養成使用它的習慣時,將會但願進行全部試驗,包括如下用法。
在最簡單的調用形式中,xargs 就像一個篩選器,它接受一個列表(每一個成員分別在單獨的行上)做爲輸入。該工具將那些成員放置在單個空格分隔的行上:
清單 14. xargs 工具產生的輸出示例
~ $ xargs
a
b
c
Control-D
a b c
~ $
您能夠發送經過 xargs 來輸出文件名的任何工具的輸出,以便爲其餘某些接受文件名做爲參數的工具得到參數列表,如如下示例所示:
清單 15. xargs 工具的使用示例
~/tmp $ ls -1 | xargs
December_Report.pdf README a archive.tar mkdirhier.sh
~/tmp $ ls -1 | xargs file
December_Report.pdf: PDF document, version 1.3
README: ASCII text
a: directory
archive.tar: POSIX tar archive
mkdirhier.sh: Bourne shell script text executable
~/tmp $
xargs 命令不僅用於傳遞文件名。您還能夠在須要將文本篩選到單個行中的任什麼時候候使用它:
清單 16. 好習慣 7 的示例:使用 xargs 工具來將文本篩選到單個行中
~/tmp $ ls -l | xargs
-rw-r--r-- 7 joe joe 12043 Jan 27 20:36 December_Report.pdf -rw-r--r-- 1 \
root root 238 Dec 03 08:19 README drwxr-xr-x 38 joe joe 354082 Nov 02 \
16:07 a -rw-r--r-- 3 joe joe 5096 Dec 14 14:26 archive.tar -rwxr-xr-x 1 \
joe joe 3239 Sep 30 12:40 mkdirhier.sh
~/tmp $
從技術上講,使用 xargs 不多遇到麻煩。缺省狀況下,文件結束字符串是下劃線 (_);若是將該字符做爲單個輸入參數來發送,則它以後的全部內容將被忽略。爲了防止這種狀況發生,可使用 -e 標誌,它在不帶參數的狀況下徹底禁用結束字符串 。
避免經過管道將 grep 發送到 wc -l 來對輸出行數計數。grep 的 -c 選項提供了對與特定模式匹配的行的計數,而且通常要比經過管道發送到 wc 更快,如如下示例所示:
清單 17. 好習慣 8 的示例:使用和不使用 grep 的行計數
~ $ time grep and tmp/a/longfile.txt | wc -l
2811
real 0m0.097s
user 0m0.006s
sys 0m0.032s
~ $ time grep -c and tmp/a/longfile.txt
2811
real 0m0.013s
user 0m0.006s
sys 0m0.005s
除了速度因素外,-c 選項仍是執行計數的好方法。對於多個文件,帶 -c 選項的 grep 返回每一個文件的單獨計數,每行一個計數,而針對 wc 的管道則提供全部文件的組合總計數。
然而,無論是否考慮速度,此示例都代表了另外一個要避免地常見錯誤。這些計數方法僅提供包含匹配模式的行數——若是那就是您要查找的結果,這沒什麼問題。可是在行中具備某個特定模式的多個實例的狀況下,這些方法沒法爲您提供實際匹配實例數量 的真實計數。歸根結底,若要對實例計數,您仍是要使用 wc 來計數。首先,使用 -o 選項(若是您的版本支持它的話)來運行 grep 命令。此選項僅 輸出匹配的模式,每行一個模式,而不輸出行自己。可是您不能將它與 -c 選項結合使用,所以要使用 wc -l 來對行計數,如如下示例所示:
清單 18. 好習慣 8 的示例:使用 grep 對模式實例計數
~ $ grep -o and tmp/a/longfile.txt | wc -l
3402
~ $
在此例中,調用 wc 要比第二次調用 grep 並插入一個虛擬模式(例如 grep -c)來對行進行匹配和計數稍快一點。
當您只但願匹配輸出行中特定字段 中的模式時,諸如 awk 等工具要優於 grep。下面通過簡化的示例演示瞭如何僅列出 12 月修改過的文件。
清單 19. 壞習慣 9 的示例:使用 grep 來查找特定字段中的模式
~/tmp $ ls -l /tmp/a/b/c | grep Dec
-rw-r--r-- 7 joe joe 12043 Jan 27 20:36 December_Report.pdf
-rw-r--r-- 1 root root 238 Dec 03 08:19 README
-rw-r--r-- 3 joe joe 5096 Dec 14 14:26 archive.tar
~/tmp $
在此示例中,grep 對行進行篩選,並輸出其修改日期和名稱中帶 Dec 的全部文件。所以,諸如 December_Report.pdf 等文件是匹配的,即便它自從一月份以來還未修改過。這可能不是您但願的結果。爲了匹配特定字段中的模式,最好使用 awk,其中的一個關係運算符對確切的字段進行匹配,如如下示例所示:
清單 20. 好習慣 9 的示例:使用 awk 來查找特定字段中的模式
~/tmp $ ls -l | awk '$6 == "Dec"'
-rw-r--r-- 3 joe joe 5096 Dec 14 14:26 archive.tar
-rw-r--r-- 1 root root 238 Dec 03 08:19 README
~/tmp $
grep 的一個常見的基本用法錯誤是經過管道將 cat 的輸出發送到 grep 以搜索單個文件的內容。這絕對是沒必要要的,純粹是浪費時間,由於諸如 grep 這樣的工具接受文件名做爲參數。您根本不須要在這種狀況下使用 cat,如如下示例所示:
清單 21. 好習慣和壞習慣 10 的示例:使用帶和不帶 cat 的 grep
~ $ time cat tmp/a/longfile.txt | grep and
2811
real 0m0.015s
user 0m0.003s
sys 0m0.013s
~ $ time grep and tmp/a/longfile.txt
2811
real 0m0.010s
user 0m0.006s
sys 0m0.004s
~ $
此錯誤存在於許多工具中。因爲大多數工具都接受使用連字符 (-) 的標準輸入做爲一個參數,所以即便使用 cat 來分散 stdin 中的多個文件,參數也一般是無效的。僅當您使用帶多個篩選選項之一的 cat 時,才真正有必要在管道前首先執行鏈接。
若是不須要在命令提示符處鍵入長的、使人費解的文件名,這是否是很棒呢?的確,您不須要這樣作。相反,您能夠配置最流行的 UNIX Shell 以使用文件名完成。該功能在各個 Shell 中的工做方式略有不一樣,所以我將向您展現如何在最流行的 Shell 中使用文件名完成。文件名完成使您能夠更快地輸入並避免錯誤。懶惰?也許吧。效率更高?固然!
經常使用首字母縮寫詞
1) MB:兆字節
2) HTTP:超文本傳輸協議
3) HTTPS:HTTP over Secure Sockets Layer
4) FTP:文件傳輸協議
5) FTPS:FTP over Secure Sockets Layer
6) LDAP:輕型目錄訪問協議
我正在運行哪一種 Shell?
若是您不知道目前使用的是哪種 Shell,會怎麼樣?雖然這個訣竅不是另外 10 個好習慣的正式組成部分,但它仍然頗有用。可使用 echo $0 或 ps -p $$ 命令顯示您正在使用的 Shell。對於我來講,運行的是 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。)在您開始鍵入文件名後,能夠按 Esc 鍵,Shell 將完成文件名,或完成儘量多的部分。例如,假設您擁有名爲 file一、file2 和 file3 的文件。若是您鍵入 f,而後按 Esc 鍵,將填充 file,而您必須鍵入 一、2 或 3 來完成相應的文件名。
Bash
Bash Shell 也提供了文件名完成,但使用 Tab 鍵代替 Esc 鍵。您在 Bash Shell 中不須要設置任何選項便可啓用文件名完成,該選項是缺省設置的。Bash 還實現了其餘功能。鍵入文件名的一部分後,按 Tab 鍵,若是有多個文件知足您的請求,而且您須要添加文本以選擇其中一個文件,那麼您能夠多按 Tab 鍵兩次,以顯示與您目前鍵入的內容相匹配的文件的列表。使用以前名爲 file一、file2 和 file3 的文件示例,首先鍵入 f。當您按一次 Tab 鍵時,Bash 完成 file;再按一次 Tab 鍵時,將展開列表 file1 file2 file3。
Korn Shell
對於 Korn Shell 用戶,文件名完成取決於 EDITOR 變量的值。若是 EDITOR 設置爲 vi,那麼您鍵入部分名稱,而後按 Esc 鍵,後跟反斜槓 (\) 字符。若是 EDITOR 設置爲 emacs,那麼您鍵入部分名稱,而後按兩次 Esc 鍵以完成文件名。
若是您爲一系列命令使用相同的文件名,會發生什麼狀況?固然,有一種快捷方式能夠快速得到您上次使用的文件名。如清單 2 所示,!$ 命令返回前一個命令使用的文件名。從文件 this-is-a-long-lunch-menu-file.txt 中搜索單詞 pickles 的出現位置。搜索結束後,使用 vi 命令來編輯 this-is-a-long-lunch-menu-file.txt 文件,而不須要從新鍵入文件名。您使用感嘆號 (!) 來訪問歷史,而後使用美圓符號 ($) 返回前一命令的最後字段。若是您反覆用到長文件名,那麼這是一個很是好的工具。
清單 2. 使用 !$ 得到前一個命令使用的文件名
$ grep pickles this-is-a-long-lunch-menu-file.txt
pastrami on rye with pickles and onions
$ vi !$
!$ 命令返回某個命令使用的上一個文件名參數。但若是某個命令使用多個文件名,而您只但願重用其中一個文件名,該如何作?!:1 操做符返回某個命令使用的第一個文件名。清單 3 中的示例顯示能夠如何將此操做符與 !$ 運算符組合使用。在第一個命令中,將一個文件從新命名爲更有意義的名稱,但爲了保持原始文件名可用,建立了一個符號連接。從新命名文件 kxp12.c 以提升可讀性,而後使用 link 命令來建立到原始文件名的符號連接,以防在其餘位置使用該文件名。!$ 操做符返回 file_system_access.c 文件名,而 !:1 操做符返回 kxp12.c 文件名,該文件名是上個命令的第一個文件名。
清單 3. 組合使用 !$ 和 !:1
$ mv kxp12.c file_system_access.c
$ ln –s !$ !:1
UNIX 支持各類目錄導航工具。最喜歡的兩款提升工做效率的工具是 pushd 和 popd。您固然瞭解 cd 命令用於更改您的當前目錄。若是您要在多個目錄中導航,但但願可以快速返回某個位置,該如何作?pushd 和 popd 命令建立一個虛擬目錄堆棧,pushd 命令用來更改您的當前目錄並將其存儲在堆棧中,而 popd 命令用來從堆棧的頂部移除目錄並使您返回該位置。您可使用 dirs 命令來顯示當前目錄堆棧,而不會壓入或彈出新目錄。清單 4 顯示如何使用 pushd 和 popd 命令在目錄樹中快速導航。
清單 4. 使用 pushd 和 popd 在目錄樹中導航
$ 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 是一個數字,您能夠向左或向右移動堆棧,如清單 5 所示。
清單 5. 旋轉目錄堆棧
$ dirs
/usr/local/bin /var /etc ~ ~
$ pushd +1
/var /etc ~ ~ /usr/local/bin
$ pushd -1
~ /usr/local/bin /var /etc ~
是否須要找出您的全部空閒磁盤空間被什麼佔用了?您可使用如下幾個工具來管理您的存儲設備。如清單 6 所示,df 命令爲您顯示每一個可用捲上已使用的塊的總數,以及空閒空間的百分比。
清單 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 命令時附帶 -size 參數。清單 7 顯示瞭如何使用 find 命令來查找大於 10MB 的文件。請注意,-size 參數以 KB 爲單位計量大小。
清單 7. 查找大於 10MB 的全部文件
$ find / -size +10000k –xdev –exec ls –lh {}\;
如下是一個簡單示例:您須要快速建立一個簡單臨時文件,但不但願啓動您的編輯器。使用帶有 > 文件重定向操做符的 cat 命令。如清單 8 所示,使用不帶文件名的 cat 命令只回顯向標準輸入鍵入的任何內容;> 重定向將該輸入捕獲到指定的文件中。請注意,您在結束鍵入時必須提供文件結束字符,一般爲 Ctrl-D。
清單 8. 快速建立臨時文件
$ cat > my_temp_file.txt
This is my temp file text
^D
$ cat my_temp_file.txt
This is my temp file text
須要執行相同操做,可是附加到現有文件而不是建立新文件。如清單 9 所示,改用 >> 操做符。>> 文件重定向操做符向現有文件附加內容。
清單 9.快速向文件附加內容
$ cat >> my_temp_file.txt
More text
^D
$ cat my_temp_file.txt
This is my temp file text
More text
curl 命令使您可使用 HTTP、HTTPS、FTP、FTPS、Gopher、DICT、TELNET、LDAP 或 FILE 協議從服務器檢索數據。如清單 10 所示,我可使用 curl 命令從美國國家氣象局瞭解我所在位置(紐約州布法羅市)的當前天氣情況。當與 grep 命令組合使用時,我能夠檢索布法羅市的天氣情況。使用 -s 命令行選項來禁止 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 命令來下載 HTTP 託管的文件。使用 -o 參數來指定保存輸出的位置。
清單 11. 使用 curl 下載 HTTP 承載的文件
$ curl -o archive.tar http://www.somesite.com/archive.tar
這實際上只是您使用 curl 命令能夠完成的操做的提示。您只需在命令提示符處鍵入 man curl 顯示 curl 命令的完整使用信息,就能夠開始瞭解更多內容。
大量 UNIX 命令使用正則表達式做爲參數。從技術角度而言,正則表達式 是表示某種模式的字符串(也就是說,由字母、數字和符號組成的字符序列),用於定義零或更長的字符串。正則表達式使用元字符(例如,星號 [*] 和問號 [?])來匹配其餘字符串的部分或所有內容。正則表達式不必定包含通配符,但通配符可使正則表達式在搜索模式和處理文件時發揮更大的做用。表 1 顯示了一些基本正則表達式序列。
表 1. 正則表達式序列
清單 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
有關命令行正則表達式的深刻描述,閱讀 developerWorks 文章"對話 UNIX,第 9 部分:正則表達式。"
有時,您可能但願肯定某個特定用戶是否運行過您的管理腳本。爲找出答案,您可使用 whoami 命令來返回當前用戶的名稱。清單 13 顯示了獨自運行的 whoami 命令;清單 14 顯示了使用 whoami 確保當前用戶不是根用戶的 Bash 腳本的摘錄。
清單 13. 從命令行使用 whoami
$ whoami
John
清單 14. 在腳本中使用 whoami
if [ $(whoami) = "root" ]
then
echo "You cannot run this script as root."
exit 1
fi
awk 命令彷佛始終處在 Perl 的陰影下,但它對於簡單、基於命令行的數據處理來講是一個快速、實用的工具。清單 15 顯示瞭如何開始使用 awk 命令。若要獲取文件中每行文本的長度,請使用 length() 函數。若要查看字符串 ing 是否出如今文件文本中,請使用 index() 函數,該函數返回 ing 首次出現的位置,這樣您就可使用它來進行進一步的字符串處理。若要 tokenize(也就是說,將一行拆分爲單詞長度的片斷)某個字符串,請使用 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
打印文本文件中的指定字段是一項簡單的 awk 任務。在清單 16 中,sales 文件包含每一個銷售人員的姓名,後跟每個月銷售數字。您可使用 awk 命令來快速得到每月的銷售總額。缺省狀況下,awk 將每一個以逗號分隔的值視爲不一樣的字段。您使用 $n 操做符來訪問每一個字段。
清單 16. 使用 awk 對數據進行彙總
$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
成爲命令行高手須要進行一些實踐。按照相同的方式處理問題很簡單,由於您已經習慣了。擴展您的命令行資源能夠顯著提升您的工做效率,並促使您朝着 UNIX 命令行高手的方向前進!