2

Я использую Mac 10.7.5. В терминале я буду использовать эту команду для поиска файлов со строками

find src/main -name "*" | xargs grep -i 'mystring'

Однако, когда файлы содержат пробелы, я получу результаты, как

grep: images/mystuff/essential-questions-selected-arrow-ela: No such file or directory
grep: 5.24.38: No such file or directory
grep: PM.png: No such file or directory

Фактический файл в приведенном выше примере - "essential-questions-selected-arrow-ela 5.24.38 PM.png". Как мне успешно выполнить указанную выше команду, даже если в искомых файлах есть пробелы?

2 ответа2

4

Если вы хотите использовать выходные данные find в xargs , рекомендуемый способ - передать их с помощью символов NUL чтобы разделить каждое имя файла:

find src/main -name "*" -print0 | xargs -0 grep -i 'mystring'

Причина этого заключается в том, что оболочка (и, в частности, xargs) обычно разделяет аргументы (или входные данные в случае xargs) на основе пробелов, то есть пробелов, табуляции и новых строк. Как только вы используете -0 , xargs будет читать каждое поле, разделенное NUL , что и выводит find -print0 .

Это будет работать в GNU find и xargs а также в версиях, включенных в OS X. Другие версии могут не иметь опций, поскольку они не требуются в POSIX.


Но опять же, это не является действительно необходимым. Шаблон имени "*" распространяется на все возможные имена. grep может выполнять поиск самостоятельно, поэтому все, что нужно, это:

grep -ri 'mystring' src/main

В Bash 4 (по умолчанию не поставляется с OS X) вы также можете выполнять рекурсивное сглаживание, например, над всеми файлами .txt , используя параметр globstar:

shopt -s globstar
grep -i 'mystring' **/*.txt
2

Безусловно, самый безопасный и простой способ сделать это - использовать опцию find -exec . От man find:

-exec utility [argument ...] ;
     True if the program named utility returns a zero value as its exit status.
     Optional arguments may be passed to the utility.  The expression must be 
     terminated by a semicolon (``;'').  If you invoke find from a shell you 
     may need to quote the semicolon if the shell would otherwise treat it as a 
     control operator.  If the string ``{}'' appears anywhere in the utility 
     name or the arguments it is replaced by the pathname of the current file.  
     Utility will be executed from the directory from which find was executed.  
     Utility and arguments are not subject to the further expansion of shell 
     patterns and constructs.

-exec utility [argument ...] {} +
     Same as -exec, except that ``{}'' is replaced with as many pathnames as 
     possible for each invo-cation invocationcation of utility. This behaviour 
     is similar to that of xargs(1).

Другими словами, опция -exec будет запускать все, что вы дадите ей по результатам поиска, заменяя {} на каждый найденный файл (или каталог). Итак, чтобы выполнить поиск определенной строки, вы должны сделать:

find src/main -name "*" -exec grep -i 'mystring' {} +

Это, однако, также найдет каталоги и выдаст ошибку. Имейте в виду, что он будет работать, он просто будет жаловаться, когда вы попытаетесь запустить его в каталоге, у вас возникла бы та же проблема с использованием xargs . То, что вы на самом деле пытаетесь сделать здесь, это найти все файлы и только файлы. В этом случае -name '*' не требуется, поскольку find src/main точно такой же, как find src/main -name "*" . Таким образом, вместо того, чтобы использовать это, укажите, что вы хотите только найти файлы:

find src/main -type f -exec grep -i 'mystring' {} +

Всё ещё ищете ответ? Посмотрите другие вопросы с метками .