web-dev-qa-db-fra.com

Trouver des fichiers contenant une chaîne dans le nom de fichier et une chaîne différente dans le fichier?

Je voudrais (récursivement) trouver tous les fichiers avec "ABC" dans leur nom de fichier, qui contiennent également "XYZ" dans le fichier. J'ai essayé:

find . -name "*ABC*" | grep -R 'XYZ'

mais cela ne donne pas la sortie correcte.

18
user997112

C'est parce que grep ne peut pas lire les noms de fichiers à rechercher à partir de l'entrée standard. Ce que vous faites, c'est imprimer le fichier names qui contient XYZ. Utilisez find-exec option à la place:

find . -name "*ABC*" -exec grep -H 'XYZ' {} +

De 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. 

[...]

   -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.

Si vous n'avez pas besoin des lignes correspondantes réelles, mais uniquement de la liste des noms de fichiers contenant au moins une occurrence de la chaîne, utilisez-la à la place:

find . -name "*ABC*" -exec grep -l 'XYZ' {} +
27
terdon

Je trouve la commande suivante la manière la plus simple:

grep -R --include="*ABC*" XYZ

ou ajoutez -i pour rechercher sans tenir compte de la casse:

grep -i -R --include="*ABC*" XYZ
2
R. Oosterholt

… | grep -R 'XYZ' n'a pas de sens. D'une part, -R 'XYZ' signifie agir récursivement sur le répertoire XYZ. D'autre part, … | grep 'XYZ' signifie rechercher le modèle XYZ dans l'entrée standard de grep. \

Sous Mac OS X ou BSD, grep traitera XYZ comme un modèle et se plaindra:

$ echo XYZ | grep -R 'XYZ'
grep: warning: recursive search of stdin
(standard input):XYZ

GNU grep ne se plaindra pas. Au contraire, il traite XYZ comme un modèle, ignore son entrée standard et recherche récursivement à partir du répertoire en cours.


Ce que vous vouliez faire était probablement

find . -name "*ABC*" | xargs grep -l 'XYZ'

… Qui est similaire à

grep -l 'XYZ' $(find . -name "*ABC*")

… Qui indiquent à grep de rechercher XYZ dans les noms de fichiers correspondants.

Notez cependant que tout espace dans les noms de fichiers entraînera la rupture de ces deux commandes. Vous pouvez utiliser xargs en toute sécurité en utilisant NUL comme délimiteur:

find . -name "*ABC*" -print0 | xargs -0 grep -l 'XYZ'

Mais la solution de @ terdon utilisant find … -exec grep -l 'XYZ' '{}' + est plus simple et meilleur.

1
200_success