考虑这个命令:
ls /mydir/*.txt | xargs chown root
目的是将所有文本文件的所有者更改mydir
为root
问题是,如果没有.txt
文件,mydir
则xargs会发出一条错误,指出没有指定路径.这是一个无害的例子,因为被抛出一个错误,但在某些情况下,如在我需要在这里使用的脚本,假设一个空白的路径是当前目录.因此,如果我/home/tom/
从那时起运行该命令,如果没有结果,ls /mydir/*.txt
并且所有文件/home/tom/
都将其所有者更改为root.
那么我如何让xargs忽略空结果呢?
对于GNU xargs
,您可以使用-r
或--no-run-if-empty
选项:
--no-run-if-empty
-r
如果标准输入不包含任何非空白,请不要运行该命令.通常,即使没有输入,命令也会运行一次.此选项是GNU扩展.
非GNU的xargs用户可以利用的-L <#lines>
,-n <#args>
,-i
,和-I
:
ls /empty_dir/ | xargs -n10 chown root # chown executed every 10 args or fewer ls /empty_dir/ | xargs -L10 chown root # chown executed every 10 lines or fewer ls /empty_dir/ | xargs -i cp {} {}.bak # every {} is replaced with the args from one input line ls /empty_dir/ | xargs -I ARG cp ARG ARG.bak # like -i, with a user-specified placeholder
请记住,xargs在空白处拆分行,但引用和转义是可用的; RTFM了解详情.
man xargs
说--no-run-if-empty
.
就xargs
你而言,你可以-r
按照建议使用,但BSD不支持它xargs
.
因此,作为解决方法,您可以传递一些额外的临时文件,例如:
find /mydir -type f -name "*.txt" -print0 | xargs -0 chown root $(mktemp)
或者将其stderr重定向到null(2> /dev/null
),例如
find /mydir -type f -name "*.txt" -print0 | xargs -0 chown root 2> /dev/null || true
另一种更好的方法是使用while
循环迭代找到的文件:
find /mydir -type f -name "*.txt" -print0 | while IFS= read -r -d '' file; do chown -v root "$file" done
另请参阅:在Mac OS X中忽略xargs的空结果
另请注意,您更改权限的方法并不好,不鼓励使用.当然你不应该解析ls
命令的输出(参见:为什么你不应该解析ls的输出).特别是当您通过root运行命令时,因为您的文件可以包含可由shell解释的特殊字符,或者想象具有空格字符的文件/
,那么结果可能很糟糕.
因此,您应该改变您的方法并使用find
命令,例如
find /mydir -type f -name "*.txt" -execdir chown root {} ';'