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
的作用在於,大多數命令(比如rm
、mkdir
、ls
)與管道一起使用時,都需要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
會新建三個子目錄,因為xargs
將one 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
是一個三行的文字檔案。我們希望對每一項命令列引數,執行兩個命令(echo
和mkdir
),使用-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 容器,這樣執行速度會快很多。
十一、參考連結
- Linux and Unix xargs command tutorial with examples, George Ornbo
- 8 Practical Examples of Linux Xargs Command for Beginners, Himanshu Arora
(完)