Differences between revisions 1 and 13 (spanning 12 versions)
Revision 1 as of 2007-05-02 22:52:21
Size: 1828
Editor: redondos
Comment:
Revision 13 as of 2014-04-22 00:13:35
Size: 1748
Editor: GreyCat
Comment: a few notes
Deletions are marked like this. Additions are marked like this.
Line 1: Line 1:
[[Anchor(faq8)]] <<Anchor(faq8)>>
Line 3: Line 3:
On most recent systems (GNU/Linux/BSD), you would use {{{grep -r pattern .}}} to search all files from the current directory (.) downward. If you are on a typical GNU or BSD system, all you need is one of these:
Line 5: Line 5:
You can use {{{find}}} if your {{{grep}}} lacks -r:
Line 7: Line 6:
    find . -type f -exec grep -l "$search" '{}' \; # Recurse and print matching lines (GNU grep):
grep -r -- "$search" .

# Recurse and print only the filenames (GNU grep):
grep -r -l -- "$search" .
Line 10: Line 13:
The {} characters will be replaced with the current file name. If your `grep` lacks a `-r` option, you can [[UsingFind|use find]] to do the recursion:
Line 12: Line 15:
This command is slower than it needs to be, because {{{find}}} will call {{{grep}}} with only one file name, resulting in many {{{grep}}} invocations (one per file). Since {{{grep}}} accepts multiple file names on the command line, {{{find}}} can be instrumented to call it with several file names at once:
Line 14: Line 16:
    find . -type f -exec grep -l "$search" '{}' \+ # Portable but slow
find . -type f -exec grep -l -- "$search" {} \;
Line 17: Line 20:
The trailing '+' character instructs {{{find}}} to call {{{grep}}} with as many file names as possible, saving processes and resulting in faster execution. This example works for POSIX {{{find}}}, e.g. with Solaris. This command is slower than it needs to be, because {{{find}}} will call {{{grep}}} with only one file name, resulting in many {{{grep}}} invocations (one per file). Since {{{grep}}} accepts multiple file names on the command line, {{{find}}} can be instructed to call it with several file names at once:
Line 19: Line 22:
GNU find uses a helper program called {{{xargs}}} for the same purpose:
Line 21: Line 23:
    find . -type f -print0 | xargs -0 grep -l "$search" # Fast, but requires a recent find
find . -type f -exec grep -l -- "$search" {} +
Line 23: Line 26:
The trailing '+' character instructs {{{find}}} to call {{{grep}}} with as many file names as possible, saving processes and resulting in faster execution. This example works for POSIX-2008 {{{find}}}, which ''most'' current operating systems have, but which may not be available on legacy systems.
Line 24: Line 28:
The {{{-print0}}} / {{{-0}}} options ensure that any file name can be processed, even ones containing blanks, TAB characters, or new-lines. Traditional Unix has a helper program called {{{xargs}}} for the same purpose:
Line 26: Line 30:
90% of the time, all you need is: {{{
# DO NOT USE THIS
find . -type f | xargs grep -l -- "$search"
}}}
However, if your filenames contain spaces, quotes or other metacharacters, this will fail catastrophically. BSD/GNU `xargs` has a `-print0` option:
Line 28: Line 36:
Have grep recurse and print the lines (GNU grep):
Line 30: Line 37:
    grep -r "$search" . find . -type f -print0 | xargs -0 grep -l -- "$search"
Line 32: Line 39:
The {{{-print0}}} / {{{-0}}} options ensure that any file name can be processed, even one containing [[BashFAQ/020|blanks, TAB characters, or newlines]].
Line 33: Line 41:
Have grep recurse and print only the names (GNU grep):
{{{
    grep -r -l "$search" .
}}}

The {{{find}}} command can be used to run arbitrary commands on every file in a directory (including sub-directories). Replace {{{grep}}} with the command of your choice. The curly braces {} will be replaced with the current file name in the case above.

(Note that they must be escaped in some shells, but not in ["BASH"].)
----
CategoryShell

How can I recursively search all files for a string?

If you are on a typical GNU or BSD system, all you need is one of these:

# Recurse and print matching lines (GNU grep):
grep -r -- "$search" .

# Recurse and print only the filenames (GNU grep):
grep -r -l -- "$search" .

If your grep lacks a -r option, you can use find to do the recursion:

# Portable but slow
find . -type f -exec grep -l -- "$search" {} \;

This command is slower than it needs to be, because find will call grep with only one file name, resulting in many grep invocations (one per file). Since grep accepts multiple file names on the command line, find can be instructed to call it with several file names at once:

# Fast, but requires a recent find
find . -type f -exec grep -l -- "$search" {} +

The trailing '+' character instructs find to call grep with as many file names as possible, saving processes and resulting in faster execution. This example works for POSIX-2008 find, which most current operating systems have, but which may not be available on legacy systems.

Traditional Unix has a helper program called xargs for the same purpose:

# DO NOT USE THIS
find . -type f | xargs grep -l -- "$search"

However, if your filenames contain spaces, quotes or other metacharacters, this will fail catastrophically. BSD/GNU xargs has a -print0 option:

find . -type f -print0 | xargs -0 grep -l -- "$search"

The -print0 / -0 options ensure that any file name can be processed, even one containing blanks, TAB characters, or newlines.


CategoryShell

BashFAQ/008 (last edited 2015-03-05 00:24:46 by izabera)