$ ls *mp3 | xargs mplayer
Playing Lemon.
File not found: 'Lemon'
Playing Tree.mp3.
File not found: 'Tree.mp3'
Exiting... (End of file)
我的命令失败了,因为“Lemon Tree.mp3”文件包含空格,所以xargs认为它是两个文件。我能让find + xargs与这样的文件名一起工作吗?
$ ls *mp3 | xargs mplayer
Playing Lemon.
File not found: 'Lemon'
Playing Tree.mp3.
File not found: 'Tree.mp3'
Exiting... (End of file)
我的命令失败了,因为“Lemon Tree.mp3”文件包含空格,所以xargs认为它是两个文件。我能让find + xargs与这样的文件名一起工作吗?
当前回答
我知道我没有直接回答xargs的问题,但值得一提的是find的-exec选项。
给定以下文件系统:
[root@localhost bokeh]# tree --charset assci bands
bands
|-- Dream\ Theater
|-- King's\ X
|-- Megadeth
`-- Rush
0 directories, 4 files
find命令可以处理Dream Theater和King’s x中的空间。因此,使用grep查找每个乐队的鼓手:
[root@localhost]# find bands/ -type f -exec grep Drums {} +
bands/Dream Theater:Drums:Mike Mangini
bands/Rush:Drums: Neil Peart
bands/King's X:Drums:Jerry Gaskill
bands/Megadeth:Drums:Dirk Verbeuren
在-exec选项中,{}代表包含路径的文件名。请注意,您不必转义它或将它放在引号中。
-exec的终止符(+和\;)之间的区别是+将尽可能多的文件名分组到一个命令行上。而\;将对每个文件名执行命令。
因此,find bands/ -type f -exec grep Drums{} +将导致:
grep Drums "bands/Dream Theater" "bands/Rush" "bands/King's X" "bands/Megadeth"
and find bands/ type f -exec grep Drums {} \;会导致:
grep Drums "bands/Dream Theater"
grep Drums "bands/Rush"
grep Drums "bands/King's X"
grep Drums "bands/Megadeth"
在grep的情况下,这会产生打印文件名或不打印文件名的副作用。
[root@localhost bokeh]# find bands/ -type f -exec grep Drums {} \;
Drums:Mike Mangini
Drums: Neil Peart
Drums:Jerry Gaskill
Drums:Dirk Verbeuren
[root@localhost bokeh]# find bands/ -type f -exec grep Drums {} +
bands/Dream Theater:Drums:Mike Mangini
bands/Rush:Drums: Neil Peart
bands/King's X:Drums:Jerry Gaskill
bands/Megadeth:Drums:Dirk Verbeuren
当然,grep的选项-h和-h将控制是否打印文件名,而不管如何调用grep。
xargs
Xargs还可以控制man文件在命令行上的格式。
Xargs默认将所有参数分组到一行。为了做同样的事情,-exec \;使用xargs -l。注意,-t选项告诉xargs在执行命令之前打印该命令。
[root@localhost bokeh]# find ./bands -type f | xargs -d '\n' -l -t grep Drums
grep Drums ./bands/Dream Theater
Drums:Mike Mangini
grep Drums ./bands/Rush
Drums: Neil Peart
grep Drums ./bands/King's X
Drums:Jerry Gaskill
grep Drums ./bands/Megadeth
Drums:Dirk Verbeuren
注意-l选项告诉xargs对每个文件名执行grep。
相对于默认值(即没有-l选项):
[root@localhost bokeh]# find ./bands -type f | xargs -d '\n' -t grep Drums
grep Drums ./bands/Dream Theater ./bands/Rush ./bands/King's X ./bands/Megadeth
./bands/Dream Theater:Drums:Mike Mangini
./bands/Rush:Drums: Neil Peart
./bands/King's X:Drums:Jerry Gaskill
./bands/Megadeth:Drums:Dirk Verbeuren
Xargs可以更好地控制命令行上的文件数量。给-l选项每个命令的最大文件数量。
[root@localhost bokeh]# find ./bands -type f | xargs -d '\n' -l2 -t grep Drums
grep Drums ./bands/Dream Theater ./bands/Rush
./bands/Dream Theater:Drums:Mike Mangini
./bands/Rush:Drums: Neil Peart
grep Drums ./bands/King's X ./bands/Megadeth
./bands/King's X:Drums:Jerry Gaskill
./bands/Megadeth:Drums:Dirk Verbeuren
[root@localhost bokeh]#
请注意,由于-l2, grep执行时有两个文件名。
其他回答
迪克。Guertin的回答[1]表明,可以转义文件名中的空格,这是这里建议的其他解决方案(例如使用空字符作为分隔符而不是空白字符)的有价值的替代方案。但它可以更简单——你真的不需要一个唯一的字符。你可以直接用sed添加转义的空格:
ls | grep ' ' | sed 's| |\\ |g' | xargs ...
此外,只有当您只想要名称中有空格的文件时,才需要使用grep。更一般地(例如,当处理一批文件时,其中一些有空格,一些没有),跳过grep:
ls | sed 's| |\\ |g' | xargs ...
然后,当然,文件名可以有其他空格而不是空格(例如,制表符):
ls | sed -r 's|[[:blank:]]|\\\1|g' | xargs ...
这假设你有一个支持-r(扩展的正则表达式)的sed,比如GNU sed或最新版本的bsd sed(例如,FreeBSD最初在FreeBSD 8之前拼写选项“-E”,并且至少为了兼容FreeBSD 11而同时支持-r和-E)。否则,您可以使用基本的正则表达式字符类括号表达式,并在[]分隔符中手动输入空格和制表符。
这可能更适合作为一个评论或编辑的答案,但目前我没有足够的声誉来评论,只能建议编辑。因为后面的形式(没有grep)改变了Dick的行为。格汀最初的回答,直接的编辑可能是不合适的。
替代解决方案可能会有所帮助……
您还可以使用Perl在行尾添加一个空字符,然后在xargs中使用-0选项。不像xargs -d '\n'(在已批准的答案中)-这适用于任何地方,包括OS X。
例如,要递归地列出(执行,移动等)可能包含空格或其他有趣字符的MPEG3文件-我会使用:
find . | grep \.mp3 | perl -ne 'chop; print "$_\0"' | xargs -0 ls
(注意:对于过滤,我更喜欢更容易记住的“| grep”语法而不是“find’s”——name参数。)
macOS 10.12版本。x (Sierra),如果你在文件名或子目录中有空格,你可以使用以下方法:
find . -name '*.swift' -exec echo '"{}"' \; |xargs wc -l
MacOS上的xargs没有-d选项,所以这个解决方案使用-0代替。
获取ls每行输出一个文件,然后将换行符转换为空值,并告诉xargs使用空值作为分隔符:
Ls -1 *mp3 | tr "\n" "\0" | xargs -0 mplayer
Ls | grep mp3 | sed -n“7p”| xargs -i mplayer {}
注意,在上面的命令中,xargs将为每个文件重新调用mplayer。这对于mplayer来说可能是不可取的,但是对于其他目标来说可能是可以的。