Как я могу выполнить определенную команду для каждого результата поиска?

49

Как выполнить определенную команду для каждого файла, найденного с помощью этой findкоманды? Для цели вопроса скажем, что я просто хотел бы удалить каждый найденный файл find.

FailedDev
источник

Ответы:

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.
Калеб
источник
Команда "find", предоставляемая Busybox, не поддерживает параметр -execdir, поэтому может потребоваться использовать один из методов pipe / xargs, упомянутых ниже.
MikeW
9

Альтернатива состоит в том, чтобы передать вывод и проанализировать его с последующими командами. Единственный безопасный способ сделать это - использовать -print0опцию, которая указывает findиспользовать нулевой символ в качестве разделителя результатов. Принимающие команды должны иметь возможность распознавать ввод с нулевым разделением. Пример:

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

Обратите внимание, что эта -0опция указывает, xargsчто входные данные следует рассматривать как разделенные нулями.

phunehehe
источник
Вы можете -execс большим количеством файлов, если вы закончите его +вместо ;. Смотрите ответ Калеба.
Кевин
@ Кевин, ты прав, я обновил ответ.
phunehehe
3

Find имеет встроенную команду удаления, если это все, что вам нужно сделать.

find . -name "*.txt" -delete

Любой найденный файл .txt будет удален с помощью команды выше.

SaultDon
источник
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 работает только в том случае, если ни одно из имен файлов и каталогов не содержит пробелов или любого из \"'; метод обратных цитат работает только в том случае, если ни одно из имен файлов и каталогов не содержит пробелов или любого из них \[?*.

IG83
источник
3
Оба эти метода потенциально очень опасны, особенно обратный. Существует множество потенциальных проблем с неэкранированными символами в именах файлов, которые могут привести к сбою этих методов.
Калеб
1
Я понимаю вашу точку зрения, но эти команды также могут использоваться вместе с другими инструментами, кроме find, поэтому я думаю, что их стоит упомянуть здесь.
IG83
Их, возможно, стоит упомянуть, но когда их не использовать, важно уточнить. Вопрос ОП здесь специально задан для обработки выходных данных find, для которых, -execкак правило, лучшее решение. Если вы хотите указать их как альтернативные, по крайней мере, объясните, как использовать их find -print0 | xargs -0для безопасной обработки разрыва имени файла, и уточните, когда следует соблюдать осторожность в отношении обратных кавычек.
Калеб
1
Кстати, добро пожаловать на сайт, я вижу, это ваш первый ответ. Извините, что перепрыгнул через все это. Важно научить людей заранее разбираться с проблемами, чтобы они не совершали ошибок, которые потом трудно обнаружить, но я до сих пор помню те дни, когда я не понимал, почему это тоже такая большая проблема, поэтому, пожалуйста, Не принимайте исправление как личное.
Калеб
3
Спасибо за прием! Конечно, нет никаких обид, вы, безусловно, правы в том, что -exec - подходящий способ справиться с этим. Я действительно начинаю видеть, насколько великолепна эта платформа, вы узнаете что-то новое даже из комментариев :)
IG83