如何为每个查找结果运行特定命令?


49

一个人如何对使用该find命令找到的每个文件运行一个特定命令?出于这个问题的目的,可以说我只想删除由找到的每个文件find

Answers:


60

编辑:虽然以下答案解释了一般情况,但我应该注意删除文件和目录是一种特殊情况。除了使用-execdir rm {} \;构造以外,还可以使用-delete,例如:

find -iname '*.txt' -delete

这处理了您可能不会考虑的一些极端情况,包括需要删除哪些顺序文件和目录以免出错。对于其他用例...

处理查找结果的运行命令的最佳方法通常-exec是对find命令使用各种选项。特别是,您应该尽可能使用-execdir它,因为它运行在找到的文件目录中,并且比其他选项更安全(从防止愚蠢的错误造成灾难的意义上来说)。

这些-exec选项后跟您要运行的命令,该命令{}表示应包含find查找文件所在的位置,并通过\;为每个文件运行一次命令或+替换{}为所有匹配项的参数列表而终止。注意,分号终止符被转义,因此外壳不认为它是导致新命令的分隔符。

假设您正在查找所有文本文件:

find -iname '*.txt' -execdir rm {} \;

这是查找手册(man find)中的相关内容:

   -exec command ;
          Execute  command;  true  if 0 status is returned.  All following
          arguments to find are taken to be arguments to the command until
          an  argument  consisting of ‘;’ is encountered.  The string ‘{}’
          is replaced by the current file name being processed  everywhere
          it occurs in the arguments to the command, not just in arguments
          where it is alone, as in some versions of find.  Both  of  these
          constructions might need to be escaped (with a ‘\’) or quoted to
          protect them from expansion by the shell.  See the EXAMPLES sec-
          tion for examples of the use of the -exec option.  The specified
          command is run once for each matched file.  The command is  exe-
          cuted  in  the starting directory.   There are unavoidable secu-
          rity problems surrounding use of the -exec  action;  you  should
          use the -execdir option instead.


   -exec command {} +
          This  variant  of the -exec action runs the specified command on
          the selected files, but the command line is built  by  appending
          each  selected file name at the end; the total number of invoca-
          tions of the command will  be  much  less  than  the  number  of
          matched  files.   The command line is built in much the same way
          that xargs builds its command lines.  Only one instance of  ‘{}’
          is  allowed  within the command.  The command is executed in the
          starting directory.


   -execdir command ;

   -execdir command {} +
          Like -exec, but the specified command is run from the  subdirec-
          tory  containing  the  matched  file,  which is not normally the
          directory in which you started find.  This a  much  more  secure
          method  for invoking commands, as it avoids race conditions dur-
          ing resolution of the paths to the matched files.  As  with  the
          -exec action, the ‘+’ form of -execdir will build a command line
          to process more than one matched file, but any given  invocation
          of command will only list files that exist in the same subdirec-
          tory.  If you use this option, you must ensure that  your  $PATH
          environment  variable  does  not  reference  ‘.’;  otherwise, an
          attacker can run any commands they like by leaving an  appropri-
          ately-named  file in a directory in which you will run -execdir.
          The same applies to having entries in $PATH which are  empty  or
          which are not absolute directory names.

Busybox提供的“ find”命令不支持-execdir选项,因此可能有必要使用下面提到的pipe / xargs方法之一。
MikeW

9

一种替代方法是通过管道传输输出并使用后续命令对其进行解析。唯一安全的方法是使用-print0选项,该选项告知find使用空字符作为结果定界符。接收命令必须具有识别以空分隔的输入的能力。例:

find /home/phunehehe -iregex '.*\.png$' -print0 | xargs -0 file

请注意,该-0选项告诉xargs您将输入视为空定界。


-exec如果以+而不是结尾,则可以包含更多文件;。请参阅caleb的答案。
凯文

@Kevin你是对的,我更新了答案。
phunehehe 2011年

3

如果您仅需要Find内置的delete命令。

find . -name "*.txt" -delete

找到的任何.txt文件都将使用上述命令删除。


2

我在寻找答案,但偶然发现了这个线程。答案给了我关于如何实现的想法。假设您要查找mediainfo所有JPEG文件中的

这将附加mediainfo ""每个匹配文件的开头和结尾(以尽可能地转义特殊字符),将其放入脚本并运行该脚本:

find . -name *.jpg | sed -e 's/^/mediainfo "/g;' | sed -e 's/$/"/g;' > foo.sh && sh foo.sh

如果您担心某些地方可能出错,则可以跳过将输出重定向到文件的过程,然后在运行脚本之前仅在终端中查看结果。


1

您可以使用xargs命令完成此操作。xargs本质上来说,针对其标准输入的每条指令运行一次命令。因此,例如,如果您需要删除.jpg目录中的所有文件,那么在命令行上的一种快速方法是:

$ find ./ -name "*.jpg" | xargs rm 

您还可以使用反引号(“ Tab”按钮上方)进行此操作(请注意,这是反引号字符,而不是单引号字符):

$ rm `find ./ -name "*.jpg"`

请注意,由于xargs外壳程序处理输入的方式不同,xargs方法仅在所涉及的文件名和目录名都不包含空格或以下任何一种时才有效\"'。仅当所涉及的文件名和目录名都不包含空格或任何空格时,backquote方法才有效\[?*


3
这两种方法都可能非常危险,尤其是反引号。文件名中未转义的字符存在许多潜在问题,可能导致这些方法中断。
Caleb

1
我明白您的意思,但是这些命令也可以与find之外的其他工具一起使用,因此我认为在这里值得一提。
IG83

它们可能值得一提,但是何时使用它们是很重要的。此处的OP问题专门要求处理的输出find,这-exec通常是一个更好的解决方案。如果要指定这些替代名称,请至少说明如何find -print0 | xargs -0用于安全文件名中断处理,并详细说明何时应注意反引号。
Caleb

1
顺便说一下,欢迎您访问该网站,我认为这是您的第一个答案。对不起,跳过它。重要的是要事先教人们解决问题的方法,这样他们就不会犯以后很难发现的错误,但是我仍然记得我不明白为什么这也是一个大问题的日子,所以请不要不要将更正视为个人。
卡莱布

3
谢谢您的欢迎!当然没有难过的感觉,您肯定是对的,因为-exec是处理此问题的适当方法。我真的开始看到这个平台有多
棒,
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.