我在目录树中寻找字符串foo=在文本文件中。在一个普通的Linux机器上,我有bash shell:

grep -ircl "foo=" *

目录中还有许多匹配“foo=”的二进制文件。由于这些结果不相关并降低了搜索速度,所以我希望grep跳过搜索这些文件(主要是JPEG和PNG图像)。我该怎么做呢?

我知道有——exclude=PATTERN和——include=PATTERN选项,但模式格式是什么?grep的手册页说:

--include=PATTERN     Recurse in directories only searching file matching PATTERN.
--exclude=PATTERN     Recurse in directories skip file matching PATTERN.

搜索grep包括,grep包括排除,grep排除和变体没有找到任何相关的

如果有更好的方法只在某些文件中进行grepping,我完全赞成;移动有问题的文件是行不通的。我不能只搜索某些目录(目录结构很混乱,到处都是东西)。此外,我不能安装任何东西,所以我必须使用常用工具(如grep或建议的find)。


当前回答

如果您只想跳过二进制文件,我建议您查看-I(大写I)选项。它忽略二进制文件。我经常使用以下命令:

grep -rI --exclude-dir="\.svn" "pattern" *

它会递归搜索,忽略二进制文件,并且不会在Subversion隐藏文件夹中查找我想要的任何模式。我在我工作的盒子上用"grepsvn"作为别名。

其他回答

Find和xargs是你的朋友。使用它们来过滤文件列表,而不是grep的——exclude

试试这样的方法

find . -not -name '*.png' -o -type f -print | xargs grep -icl "foo="

习惯这一点的好处是,它可以扩展到其他用例,例如计算所有非png文件中的行数:

find . -not -name '*.png' -o -type f -print | xargs wc -l

删除所有非png文件。

find . -not -name '*.png' -o -type f -print | xargs rm

etc.

正如评论中指出的,如果某些文件的名称中可能有空格,请使用-print0和xargs -0代替。

适用于TCSH .alias文件:

alias gisrc 'grep -I -r -i --exclude="*\.svn*" --include="*\."{mm,m,h,cc,c} \!* *'

花了我一段时间才弄清楚{mm,m,h,cc,c}部分不应该在引号内。 ~基斯

如果非递归搜索,则可以使用glop模式来匹配文件名。

grep "foo" *.{html,txt}

包括HTML和txt。它只在当前目录中搜索。

在子目录中搜索:

   grep "foo" */*.{html,txt}

在子目录中:

   grep "foo" */*/*.{html,txt}

GNU grep的——binary-files=without-match选项使其跳过二进制文件。(相当于其他地方提到的-I开关。)

(这可能需要最新版本的grep;至少2.5.3版本有。)

在grep 2.5.1中,您必须将这一行添加到~/。Bashrc或~/。bash配置文件

export GREP_OPTIONS="--exclude=\*.svn\*"