xargs 命令教程

阮一峰發表於2019-08-08

xargs是 Unix 系統的一個很有用的命令,但是常常被忽視,很多人不瞭解它的用法。

本文介紹如何使用這個命令。

一、標準輸入與管道命令

Unix 命令都帶有引數,有些命令可以接受"標準輸入"(stdin)作為引數。


$ cat /etc/passwd | grep root

上面的程式碼使用了管道命令(|)。管道命令的作用,是將左側命令(cat /etc/passwd)的標準輸出轉換為標準輸入,提供給右側命令(grep root)作為引數。

因為grep命令可以接受標準輸入作為引數,所以上面的程式碼等同於下面的程式碼。


$ grep root /etc/passwd

但是,大多數命令都不接受標準輸入作為引數,只能直接在命令列輸入引數,這導致無法用管道命令傳遞引數。舉例來說,echo命令就不接受管道傳參。


$ echo "hello world" | echo

上面的程式碼不會有輸出。因為管道右側的echo不接受管道傳來的標準輸入作為引數。

二、xargs 命令的作用

xargs命令的作用,是將標準輸入轉為命令列引數。


$ echo "hello world" | xargs echo
hello world

上面的程式碼將管道左側的標準輸入,轉為命令列引數hello world,傳給第二個echo命令。

xargs命令的格式如下。


$ xargs [-options] [command]

真正執行的命令,緊跟在xargs後面,接受xargs傳來的引數。

xargs的作用在於,大多數命令(比如rmmkdirls)與管道一起使用時,都需要xargs將標準輸入轉為命令列引數。


$ echo "one two three" | xargs mkdir

上面的程式碼等同於mkdir one two three。如果不加xargs就會報錯,提示mkdir缺少操作引數。

三、xargs 的單獨使用

xargs後面的命令預設是echo


$ xargs
# 等同於
$ xargs echo

大多數時候,xargs命令都是跟管道一起使用的。但是,它也可以單獨使用。

輸入xargs按下回車以後,命令列就會等待使用者輸入,作為標準輸入。你可以輸入任意內容,然後按下Ctrl d,表示輸入結束,這時echo命令就會把前面的輸入列印出來。


$ xargs
hello (Ctrl + d)
hello

再看一個例子。


$ xargs find -name
"*.txt"
./foo.txt
./hello.txt

上面的例子輸入xargs find -name以後,命令列會等待使用者輸入所要搜尋的檔案。使用者輸入"*.txt",表示搜尋當前目錄下的所有 TXT 檔案,然後按下Ctrl d,表示輸入結束。這時就相當執行find -name *.txt

四、-d 引數與分隔符

預設情況下,xargs將換行符和空格作為分隔符,把標準輸入分解成一個個命令列引數。


$ echo "one two three" | xargs mkdir

上面程式碼中,mkdir會新建三個子目錄,因為xargsone two three分解成三個命令列引數,執行mkdir one two three

-d引數可以更改分隔符。


$ echo -e "a\tb\tc" | xargs -d "\t" echo
a b c

上面的命令指定製表符\t作為分隔符,所以a\tb\tc就轉換成了三個命令列引數。echo命令的-e參數列示解釋轉義字元。

五、-p 引數,-t 引數

使用xargs命令以後,由於存在轉換引數過程,有時需要確認一下到底執行的是什麼命令。

-p引數列印出要執行的命令,詢問使用者是否要執行。


$ echo 'one two three' | xargs -p touch
touch one two three ?...

上面的命令執行以後,會列印出最終要執行的命令,讓使用者確認。使用者按下回車以後,才會真正執行。

-t引數則是列印出最終要執行的命令,然後直接執行,不需要使用者確認。


$ echo 'one two three' | xargs -t rm
rm one two three

六、-0 引數與 find 命令

由於xargs預設將空格作為分隔符,所以不太適合處理檔名,因為檔名可能包含空格。

find命令有一個特別的引數-print0,指定輸出的檔案列表以null分隔。然後,xargs命令的-0參數列示用null當作分隔符。


$ find /path -type f -print0 | xargs -0 rm

上面命令刪除/path路徑下的所有檔案。由於分隔符是null,所以處理包含空格的檔名,也不會報錯。

還有一個原因,使得xargs特別適合find命令。有些命令(比如rm)一旦引數過多會報錯"引數列表過長",而無法執行,改用xargs就沒有這個問題,因為它對每個引數執行一次命令。


$ find . -name "*.txt" | xargs grep "abc"

上面命令找出所有 TXT 檔案以後,對每個檔案搜尋一次是否包含字串abc

七、-L 引數

如果標準輸入包含多行,-L引數指定多少行作為一個命令列引數。


$ xargs find -name
"*.txt"   
"*.md"
find: paths must precede expression: `*.md'

上面命令同時將"*.txt"*.md兩行作為命令列引數,傳給find命令導致報錯。

使用-L引數,指定每行作為一個命令列引數,就不會報錯。


$ xargs -L 1 find -name
"*.txt"
./foo.txt
./hello.txt
"*.md"
./README.md

上面命令指定了每一行(-L 1)作為命令列引數,分別執行一次命令(find -name)。

下面是另一個例子。


$ echo -e "a\nb\nc" | xargs -L 1 echo
a
b
c

上面程式碼指定每行執行一次echo命令,所以echo命令執行了三次,輸出了三行。

八、-n 引數

-L引數雖然解決了多行的問題,但是有時使用者會在同一行輸入多項。


$ xargs find -name
"*.txt" "*.md"
find: paths must precede expression: `*.md'

上面的命令將同一行的兩項作為命令列引數,導致報錯。

-n引數指定每次將多少項,作為命令列引數。


$ xargs -n 1 find -name

上面命令指定將每一項(-n 1)標準輸入作為命令列引數,分別執行一次命令(find -name)。

下面是另一個例子。


$ echo {0..9} | xargs -n 2 echo
0 1
2 3
4 5
6 7
8 9

上面命令指定,每兩個引數執行一次echo命令。所以,10個阿拉伯數字執行了五次echo命令,輸出了五行。

九、-I 引數

如果xargs要將命令列引數傳給多個命令,可以使用-I引數。

-I指定每一項命令列引數的替代字串。


$ cat foo.txt
one
two
three

$ cat foo.txt | xargs -I file sh -c 'echo file; mkdir file'
one 
two
three

$ ls 
one two three

上面程式碼中,foo.txt是一個三行的文字檔案。我們希望對每一項命令列引數,執行兩個命令(echomkdir),使用-I file表示file是命令列引數的替代字串。執行命令時,具體的引數會替代掉echo file; mkdir file裡面的兩個file

十、--max-procs 引數

xargs預設只用一個程式執行命令。如果命令要執行多次,必須等上一次執行完,才能執行下一次。

--max-procs引數指定同時用多少個程式並行執行命令。--max-procs 2表示同時最多使用兩個程式,--max-procs 0表示不限制程式數。


$ docker ps -q | xargs -n 1 --max-procs 0 docker kill

上面命令表示,同時關閉儘可能多的 Docker 容器,這樣執行速度會快很多。

十一、參考連結

(完)

相關文章