xargs 命令教程筆記

一、標準輸入與管道命令

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

$ cat /etc/passwd | grep root

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

因爲grep命令可以接受標準輸入作爲參數,所以上面的代碼等同於下面的代碼。

grep root /etc/passwd

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

$ echo "hello world" | 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 ?...

上面的命令執行以後,會打印出最終要執行的命令,讓用戶確認。用戶輸入y以後(大小寫皆可),纔會真正執行。

-t參數則是打印出最終要執行的命令,然後直接執行,不需要用戶確認。

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

六、-o參數與 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 容器,這樣運行速度會快很多。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章