2021-05-12 14:32:11
Linux命令列的藝術
2020-06-16 17:50:02
熟練使用命令列是一種常常被忽視或被認為難以掌握的技能,但實際上,它可以提高你作為工程師的靈活性以及生產力。本文是一份我在 Linux 上工作時發現的一些關於命令列的使用的小技巧的摘要。有些小技巧是非常基礎的,而另外一些則是相當複雜的甚至晦澀難懂的。這篇文章並不長,但當你能夠熟練掌握這裡列出的所有技巧時,你就學會了很多關於命令列的東西了。
必讀
涵蓋範圍:
- 這篇文章對剛接觸命令列的新手以及具有命令列使用經驗的人都有用處。本文致力於做到覆蓋面廣(盡量包括一切重要的內容),具體(給出最常見的具體的例子)以及簡潔(避免一些不必要的東西以及一些偏題的可以在其他地方翻閱到文獻的東西)。 每個小技巧在某個特定情境下都是基本的或能夠顯著地節約時間。
- 本文為 Linux 所寫,除了僅限 MacOS X 系統節。其它節中的大部分內容都適用於其它 Unix 系統或 MacOS 系統,甚至 Cygwin。
- 本文關注於互動式 Bash,儘管很多技巧也適用於其他 shell 或 Bash 指令碼。
- 本文包括了“標準的”Unix 命令和需要安裝特定包的命令,只要它們足夠重要。
注意事項:
- 為了能在一頁內展示儘量多的東西,一些具體的資訊會被間接的包含在參照頁裡。聰明機智的你如果掌握了使用 Google 搜尋引擎的基本思路與命令,那麼你將可以查閱到更多的詳細資訊。使用
apt-get
/yum
/dnf
/pacman
/pip
/brew
(以及其它合適的包管理器)來安裝新程式。 - 使用 Explainshell 去獲取相關命令、引數、管道等內容的解釋。
基礎
- 學習 Bash 的基礎知識。具體來說,輸入
man bash
並至少全文瀏覽一遍; 它很簡單並且不長。其他的 shell 可能很好用,但 Bash 功能強大且幾乎所有情況下都是可用的 ( 只學習 zsh,fish 或其他的 shell 的話,在你自己的電腦上會顯得很方便,但在很多情況下會限制你,比如當你需要在伺服器上工作時)。 - 學習並掌握至少一個基於文字的編輯器。通常 Vim (
vi
) 會是你最好的選擇,因為在終端裡進行隨機編輯 Vim 真的毫無敵手,哪怕是 Emacs、某大型 IDE 甚至時下非常流行的編輯器。 - 學會如何使用
man
命令去閱讀文件。學會使用apropos
去查詢文件。了解有些命令並不對應可執行檔案,而是Bash內建的,可以使用help
和help -d
命令獲取幫助資訊。 - 學會使用
>
和<
來重定向輸出和輸入,學會使用|
來重定向管道。明白>
會覆蓋了輸出檔案而>>
是在檔案未新增。了解標準輸出 stdout 和標準錯誤 stderr。 - 學會使用萬用字元
*
(或許再算上?
和[
…]
) 和參照以及參照中'
和"
的區別。 - 熟悉 Bash 任務管理工具:
&
,ctrl-z,ctrl-c,jobs
,fg
,bg
,kill
等。 - 了解
ssh
,以及學會通過使用ssh-agent
,ssh-add
等命令來實現基本的無密碼認證。 - 學會基本的檔案管理:
ls
和ls -l
(了解ls -l
中每一列代表的意義),less
,head
,tail
和tail -f
(甚至less +F
),ln
和ln -s
(了解硬連結與軟連結的區別),chown
,chmod
,du
(硬碟使用情況概述:du -hk *
)。 關於檔案系統的管理,學習df
,mount
,fdisk
,mkfs
,lsblk
。知道 inode 是什麼(與ls -i
和df -i
等命令相關)。 - 學習基本的網路管理:
ip
或ifconfig
,dig
。 - 熟悉正規表示式,以及
grep
/egrep
裡不同引數的作用,例如-i
,-o
,-v
,-A
,-B
和-C
,這些引數是值得學習並掌握的。 - 學會使用
apt-get
,yum
,dnf
或pacman
(取決於你使用的 Linux 發行版)來查詢或安裝軟體包。並確保你的環境中有pip
來安裝基於 Python 的命令列工具 (接下來提到的部分程式使用pip
來安裝會很方便)。
日常使用
- 在 Bash 中,可以使用 Tab 自動補全引數,使用 ctrl-r 搜尋命令列歷史。
- 在 Bash 中,可以使用 ctrl-w 刪除你鍵入的最後一個單詞,使用 ctrl-u 刪除整行,使用 alt-b 和 alt-f 以單詞為單位移動游標,使用 ctrl-a 將游標移至行首,使用 ctrl-e 將游標移至行尾,使用 ctrl-k 刪除游標至行尾的所有內容,使用ctrl-l 清屏。鍵入
man readline
檢視 Bash 中的預設快捷鍵,內容很多。例如 alt-. 迴圈地移向前一個引數,以及 alt-* 展開萬用字元。 - 你喜歡的話,可以鍵入
set -o vi
來使用 vi 風格的快捷鍵,而set -o emacs
可以把它改回來。 - 為了方便地鍵入長命令,在設定你的編輯器後(例如
export EDITOR=vim
),鍵入 ctrl-x ctrl-e 會開啟一個編輯器來編輯當前命令。在 vi 模式下則鍵入 escape-v 實現相同的功能。 - 鍵入
history
檢視命令列歷史記錄。其中有許多縮寫,例如!$
(最後鍵入的引數)和!!
(最後鍵入的命令),儘管通常被 ctrl-r 和 alt-. 取代。 - 回到上一個工作路徑:
cd -
- 如果你輸入命令的時候改變了主意,按下 alt-# 來在行首新增
#
,或者依次按下 ctrl-a, #, enter。這樣做的話,之後你可以很方便的利用命令列歷史回到你剛才輸入到一半的命令。 - 使用
xargs
( 或parallel
)。他們非常給力。注意到你可以控制每行引數個數(-L
)和最大併行數(-P
)。如果你不確定它們是否會按你想的那樣工作,先使用xargs echo
檢視一下。此外,使用-I{}
會很方便。例如:
pstree -p
有助於展示進程樹。- 使用
pgrep
和pkill
根據名字查詢進程或傳送信號(-f
引數通常有用)。 - 了解你可以發往進程的信號的種類。比如,使用
kill -STOP [pid]
停止一個進程。使用man 7 signal
檢視詳細列表。 - 使用
nohup
或disown
使一個後台進程持續執行。 - 使用
netstat -lntp
或ss -plat
檢查哪些進程在監聽埠(預設是檢查 TCP 埠; 使用引數-u
檢查 UDP 埠)。 - 有關開啟通訊端和檔案,請參閱
lsof
。 - 使用
uptime
或w
來檢視系統已經執行多長時間。 - 使用
alias
來建立常用命令的快捷形式。例如:alias ll='ls -latr'
使你可以方便地執行ls -latr
命令。 - 在 Bash 指令碼中,使用
set -x
去偵錯輸出,儘可能的使用嚴格模式,使用set -e
令指令碼在發生錯誤時退出而不是繼續執行,使用set -u
來檢查是否使用了未賦值的變數,使用set -o pipefail
嚴謹地對待錯誤(儘管問題可能很微妙)。當牽扯到很多指令碼時,使用trap
。一個好的習慣是在指令碼檔案開頭這樣寫,這會使它檢測一些錯誤,並在錯誤發生時中斷程式並輸出資訊:
- 在 Bash 指令碼中,子 shell(使用括號
(...)
)是一種便捷的方式去組織引數。一個常見的例子是臨時地移動工作路徑,程式碼如下:
- 在 Bash 中,注意到其中有許多形式的擴充套件。檢查變數是否存在:
${name:?error message}
。例如,當 Bash 指令碼需要一個引數時,可以使用這樣的程式碼input_file=${1:?usage: $0 input_file}
。數學表示式:i=$(( (i + 1) % 5 ))
。序列:{1..10}
。 截斷字串:${var%suffix}
和${var#prefix}
。例如,假設var=foo.pdf
,那麼echo ${var%.pdf}.txt
將輸出foo.txt
。 - 使用括號擴充套件(
{
…}
)可以減少不必要的重複輸入相似文字以及文字組合。 - 通過使用
<(some command)
可以將輸出視為檔案。例如,對比本地檔案/etc/hosts
和一個遠端檔案:
- 了解 Bash 中的“here documents”,例如
cat <<EOF ...
。 - 在 Bash 中,同時重定向標準輸出和標準錯誤,
some-command >logfile 2>&1
。通常,為了保證命令不會在標準輸入裡殘留一個開啟了的檔案控制代碼導致你當前所在的終端無法操作,新增</dev/null
是一個好習慣。 - 使用
man ascii
檢視具有十六進位制和十進位制值的ASCII表。man unicode
,man utf-8
,以及man latin1
有助於你去了解通用的編碼資訊。 - 使用
screen
或tmux
來使用多個螢幕,當你在使用 ssh 時(儲存 session 資訊)將尤為有用。另一個輕量級的解決方案是dtach
。 - ssh 中,了解如何使用
-L
或-D
(偶爾需要用-R
)去開啟隧道是非常有用的,例如當你需要從一台遠端伺服器上存取 web。 - 對 ssh 設定做一些小優化可能是很有用的,例如這個
~/.ssh/config
檔案包含了防止特定環境下斷開連線、壓縮資料、多通道等選項:
- 部分其他的關於 ssh 的選項是安全敏感且應當小心啟用的。例如在可信任的網路中:
StrictHostKeyChecking=no
,ForwardAgent=yes
- 考慮使用
mosh
作為 ssh 的替代品,它使用 UDP 協定。 - 獲取檔案的八進位制格式許可權,使用類似如下的程式碼:
- 使用
percol
或者fzf
可以互動式地從另一個命令輸出中選取值。 - 使用
fpp
(PathPicker)可以與基於另一個命令(例如git
)輸出的檔案互動。 - 將 web 伺服器上當前目錄下所有的檔案(以及子目錄)暴露給你所處網路的所有使用者,使用:
python -m SimpleHTTPServer 7777
(使用埠 7777 和 Python 2)或python -m http.server 7777
(使用埠 7777 和 Python 3)。 - 以某種許可權執行命令,使用
sudo
(root 許可權)或sudo -u
(其他使用者)。使用su
或者sudo bash
來啟動一個以對應使用者許可權執行的 shell。使用su -
模擬其他使用者的登入。
檔案及資料處理
- 在當前路徑下通過檔名定位一個檔案,
find . -iname '*something*'
(或類似的)。在所有路徑下通過檔名查詢檔案,使用locate something
(但請記住updatedb
可能沒有對最近新建的檔案建立索引)。 - 使用
ag
在原始碼或資料檔案裡檢索(比grep -r
更好)。 - 將HTML轉為文字:
lynx -dump -stdin
- Markdown,HTML,以及所有文件格式之間的轉換,試試
pandoc
。 - 如果你不得不處理 XML,
xmlstarlet
寶刀未老。 - 使用
jq
處理 JSON。 - Excel 或 CSV 檔案的處理,csvkit 提供了
in2csv
,csvcut
,csvjoin
,csvgrep
等工具。 - 關於 Amazon S3,
s3cmd
很方便而s4cmd
更快。Amazon 官方的aws
以及saws
是其他 AWS 相關工作的基礎。 - 了解如何使用
sort
和uniq
,包括 uniq 的-u
引數和-d
引數,詳見後文一行程式碼節。另外可以了解一下comm
。 - 了解如何使用
cut
,paste
和join
來更改檔案。很多人都會使用cut
,但幾乎都不會使用join
。 - 了解如何運用
wc
去計算新行數(-l
),字元數(-m
),單詞數(-w
)以及位元組數(-c
)。 - 了解如何使用
tee
將標準輸入複製到檔案甚至標準輸出,例如ls -al | tee file.txt
。 - 了解語言環境對許多命令列工具的微妙影響,包括排序的順序和效能。大多數 Linux 的安裝過程會將
LANG
或其他有關的變數設定為符合原生的設定。意識到當你改變語言環境時,排序的結果可能會改變。明白國際化可能會時 sort 或其他命令執行效率下降許多倍。某些情況下(例如集合運算)你可以放心的使用export LC_ALL=C
來忽略掉國際化並使用基於位元組的順序。 - 了解
awk
和sed
關於資料的簡單處理的用法。例如,將文字檔案中第三列的所有數位求和:awk '{ x += $3 } END { print x }'
. 這可能比同等作用的 Python 程式碼快三倍且程式碼量少三倍。 - 替換一個或多個檔案中出現的字串:
- 依據某種模式批次重新命名多個檔案,使用
rename
。對於複雜的重新命名規則,repren
或許有幫助。
- 使用
shuf
從一個檔案中隨機選取多行。 - 了解
sort
的引數。處理數位方面,使用-n
或者-h
來處理可讀性數位(例如du -h
的輸出)。明白鍵的工作原理(-t
和-k
)。例如,注意到你需要-k1,1
來僅按第一個域來排序,而-k1
意味著按整行排序。穩定排序(sort -s
)在某些情況下很有用。例如,以第二個域為主關鍵字,第一個域為次關鍵字進行排序,你可以使用sort -k1,1 | sort -s -k2,2
。 - 如果你想在 Bash 命令列中寫 tab 製表符,按下 ctrl-v [Tab] 或鍵入
$'t'
(後者可能更好,因為你可以複製貼上它)。 - 標準的原始碼對比及合併工具是
diff
和patch
。使用diffstat
檢視變更總覽資料。注意到diff -r
對整個資料夾有效。使用diff -r tree1 tree2 | diffstat
檢視變更總覽資料。 - 對於二進位制檔案,使用
hd
使其以十六進位制顯示以及使用bvi
來編輯二進位制。 - 同樣對於二進位制檔案,使用
strings
(包括grep
等等)允許你查詢一些文字。 - 二進位制檔案對比(Delta 壓縮),使用
xdelta3
。 - 使用
iconv
更改文字編碼。而更高階的用法,可以使用uconv
,它支援一些高階的 Unicode 功能。例如,這條命令將所有元音字母轉為小寫並移除了:
- 拆分檔案,檢視
split
(按大小拆分)和csplit
(按模式拆分)。 - 使用
zless
,zmore
,zcat
和zgrep
對壓縮過的檔案進行操作。
系統偵錯
curl
和curl -I
可以便捷地被應用於 web 偵錯中,它們的好兄弟wget
也可以,或者是更潮的httpie
。- 使用
iostat
、netstat
、top
(htop
更佳)和dstat
去獲取硬碟、cpu 和網路的狀態。熟練掌握這些工具可以使你快速的對系統的當前狀態有一個大概的認識。 - 使用
netstat
和ss
檢視網路連線的細節。 - 若要對系統有一個深度的總體認識,使用
glances
。它在一個終端視窗中向你提供一些系統級的資料。這對於快速的檢查各個子系統非常有幫助。 - 若要了解記憶體狀態,執行並理解
free
和vmstat
的輸出。尤其注意“cached”的值,它指的是 Linux 核心用來作為檔案快取的記憶體大小,因此它與空閒記憶體無關。 - Java 系統偵錯則是一件截然不同的事,一個可以用於 Oracle 的 JVM 或其他 JVM 上的偵錯的小技巧是你可以執行
kill -3 <pid>
同時一個完整的棧軌跡和堆概述(包括 GC 的細節)會被儲存到標準輸出/紀錄檔檔案。JDK 中的jps
,jstat
,jstack
,jmap
很有用。SJK tools 更高階. - 使用
mtr
去跟蹤路由,用於確定網路問題。 - 用
ncdu
來檢視磁碟使用情況,它比常用的命令,如du -sh *
,更節省時間。 - 查詢正在使用頻寬的通訊端連線或進程,使用
iftop
或nethogs
。 ab
工具(綑綁於 Apache)可以簡單粗暴地檢查 web 伺服器的效能。對於更複雜的負載測試,使用siege
。wireshark
,tshark
和ngrep
可用於複雜的網路偵錯。- 了解
strace
和ltrace
。這倆工具在你的程式執行失敗、掛起甚至崩潰,而你卻不知道為什麼或你想對效能有個總體的認識的時候是非常有用的。注意 profile 引數(-c
)和附加到一個執行的進程引數 (-p
)。 - 了解使用
ldd
來檢查共用庫。 - 了解如何運用
gdb
連線到一個執行著的進程並獲取它的堆疊軌跡。 - 學會使用
/proc
。它在偵錯正在出現的問題的時候有時會效果驚人。比如:/proc/cpuinfo
,/proc/meminfo
,/proc/cmdline
,/proc/xxx/cwd
,/proc/xxx/exe
,/proc/xxx/fd/
,/proc/xxx/smaps
(這裡的xxx
表示進程的 id 或 pid)。 - 當偵錯一些之前出現的問題的時候,
sar
非常有用。它展示了 cpu、記憶體以及網路等的歷史資料。 - 關於更深層次的系統分析以及效能分析,看看
stap
(SystemTap),perf
,以及sysdig
。 - 檢視你當前使用的系統,使用
uname
,uname -a
(Unix/kernel 資訊) 或者lsb_release -a
(Linux 發行版資訊)。 - 無論什麼東西工作得很歡樂時試試
dmesg
(可能是硬體或驅動問題)。
一行程式碼
一些命令組合的例子:
- 當你需要對文字檔案做集合交、並、差運算時,結合使用
sort
/uniq
很有幫助。假設a
與b
是兩內容不同的檔案。這種方式效率很高,並且在小檔案和上G的檔案上都能運用 (sort
不被記憶體大小約束,儘管在/tmp
在一個小的根分割區上時你可能需要-T
引數),參閱前文中關於LC_ALL
和sort
的-u
引數的部分。
- 使用
grep . *
來閱讀檢查目錄下所有檔案的內容,例如檢查一個充滿組態檔的目錄比如/sys
、/proc
、/etc
。 - 計算文字檔案第三列中所有數的和(可能比同等作用的 Python 程式碼快三倍且程式碼量少三倍):
- 如果你想在檔案樹上檢視大小日期,這可能看起來像遞回版的
ls -l
但比ls -lR
更易於理解:
- 假設你有一個類似於 web 伺服器紀錄檔檔案的文字檔案,並且一個確定的值只會出現在某些行上,假設一個
acct_id
引數在URI中。如果你想計算出每個acct_id
值有多少次請求,使用如下程式碼:
- 執行這個函數從這篇文件中隨機獲取一條小技巧(解析 Markdown 檔案並抽取專案):
冷門但有用
expr
:計算表示式或正則匹配m4
:簡單地宏處理器yes
:多次列印字串cal
:漂亮的日曆env
:執行一個命令(指令碼檔案中很有用)printenv
:列印環境變數(偵錯時或在使用指令碼檔案時很有用)look
:查詢以特定字串開頭的單詞cut
、paste
和join
:資料修改fmt
:格式化文字段落pr
:將文字格式化成頁/列形式fold
:包裹文字中的幾行column
:將文字格式化成多列或表格expand
和unexpand
:製表符與空格之間轉換nl
:新增行號seq
:列印數位bc
:計算器factor
:分解因數gpg
:加密並簽名檔案toe
:terminfo entries 列表nc
:網路偵錯及資料傳輸socat
:通訊端代理,與netcat
類似slurm
:網路視覺化dd
:檔案或裝置間傳輸資料file
:確定檔案型別tree
:以樹的形式顯示路徑和檔案,類似於遞回的ls
stat
:檔案資訊time
:執行命令,並計算執行時間lockfile
:使檔案只能通過rm -f
移除logrotate
: 切換、壓縮以及傳送紀錄檔檔案watch
:重複執行同一個命令,展示結果並高亮有更改的部分tac
:反向輸出檔案shuf
:檔案中隨機選取幾行comm
:一行一行的比較排序過的檔案pv
:監視通過管道的資料hd
,hexdump
,xxd
,biew
和bvi
:儲存或編輯二進位制檔案strings
:從二進位制檔案中抽取文字tr
:轉換字母iconv
或uconv
:簡易的檔案編碼split
和csplit
:分割檔案sponge
:在寫入前讀取所有輸入,在讀取檔案後再向同一檔案寫入時比較有用,例如grep -v something some-file | sponge some-file
units
:將一種計量單位轉換為另一種等效的計量單位(參閱/usr/share/units/definitions.units
)apg
:隨機生成密碼7z
:高比例的檔案壓縮ldd
:動態庫資訊nm
:提取 obj 檔案中的符號ab
:效能分析 web 伺服器strace
:系統呼叫偵錯mtr
:更好的網路偵錯跟蹤工具cssh
:視覺化的併行 shellrsync
:通過 ssh 或本地檔案系統同步檔案和資料夾wireshark
和tshark
:抓包和網路偵錯工具ngrep
:網路層的 grephost
和dig
:DNS 查詢lsof
:列出當前系統開啟檔案的工具以及檢視埠資訊dstat
:系統狀態檢視glances
:高層次的多子系統總覽iostat
:硬碟使用狀態mpstat
: CPU 使用狀態vmstat
: 記憶體使用狀態htop
:top 的加強版last
:登入記錄w
:檢視處於登入狀態的使用者id
:使用者/組 ID 資訊sar
:系統歷史資料iftop
或nethogs
:通訊端及進程的網路利用ss
:通訊端資料dmesg
:引導及系統錯誤資訊sysctl
: 在核心執行時動態地檢視和修改核心的執行引數hdparm
:SATA/ATA 磁碟更改及效能分析lsb_release
:Linux 發行版資訊lsblk
:列出塊裝置資訊:以樹形展示你的磁碟以及磁碟分割區資訊lshw
,lscpu
,lspci
,lsusb
和dmidecode
:檢視硬體資訊,包括 CPU、BIOS、RAID、顯示卡、USB裝置等lsmod
和modinfo
:列出核心模組,並顯示其細節fortune
,ddate
和sl
:額,這主要取決於你是否認為蒸汽火車和莫名其妙的名人名言是否“有用”
僅限 MacOS X 系統
以下是僅限於 MacOS 系統的技巧
- 用
brew
(Homebrew)或者port
(MacPorts)進行包管理。這些可以用來在 Mac 系統上安裝以上的大多數命令。 - 用
pbcopy
複製任何命令的輸出到桌面應用,用pbpaste
貼上輸入。 - 在終端中將 Option 鍵視為 alt 鍵,Preferences -> Profiles -> Keyboard 勾選上 “Use Option as Meta key”。
- 用
open
或者open -a /Applications/Whatever.app
使用桌面應用開啟檔案。 - Spotlight: 用
mdfind
搜尋檔案,用mdls
列出後設資料(例如照片的 EXIF 資訊)。 - 注意 MacOS 系統是基於 BSD UNIX 的,許多命令(例如
ps
,ls
,tail
,awk
,sed
)都和 Linux 中有些微的不同,這些極大的被 System V-style Unix 和 GNU 工具影響。你可以通過標題為 “BSD General Commands Manual” 的 man 頁面發現這些不同。在有些情況下 GNU 版本的命令也可能被安裝(例如gawk
和gsed
對應 GNU 中的 awk 和 sed )。如果要寫跨平台的 Bash 指令碼,避免使用這些命令(例如,考慮 Python 或者perl
)或者經過仔細的測試。
本文永久更新連結地址:http://www.linuxidc.com/Linux/2015-11/124989.htm
相關文章