我如何包括管道在我的linux find -exec命令中?

时间:2008-11-20 22:01:52

标签: linux command-line

这不起作用。可以在查找中完成吗?或者我需要xargs吗?

find -name 'file_*' -follow -type f -exec zcat {} \| agrep -dEOE 'grep' \;

6 个答案:

答案 0 :(得分:238)

解决方案很简单:通过sh

执行
... -exec sh -c "zcat {} | agrep -dEOE 'grep' " \;

答案 1 :(得分:130)

解释管道符号作为运行多个进程的指令并将一个进程的输出传递到另一个进程的输入的工作是shell的责任(/ bin / sh或等效的)。

在您的示例中,您可以选择使用顶级shell来执行管道,如下所示:

find -name 'file_*' -follow -type f -exec zcat {} \; | agrep -dEOE 'grep'

在效率方面,这个结果需要一次调用find,多次调用zcat,以及一次调用agrep。

这将导致只生成一个agrep进程,该进程将处理由多次zcat调用产生的所有输出。

如果您出于某种原因想多次调用agrep,您可以这样做:

find . -name 'file_*' -follow -type f \
    -printf "zcat %p | agrep -dEOE 'grep'\n" | sh

这构造了一个使用管道执行的命令列表,然后将这些命令发送到新shell以实际执行。 (省略最后的“| sh”是调试或执行这样的命令行干运行的好方法。)

在效率方面,这个结果需要一次调用find,一次调用sh,多次调用zcat和多次调用agrep。

在命令调用次数方面最有效的解决方案是Paul Tomblin的建议:

find . -name "file_*" -follow -type f -print0 | xargs -0 zcat | agrep -dEOE 'grep'

...这需要一次find调用,一次xargs调用,一些zcat调用和一次agrep调用。

答案 2 :(得分:16)

find . -name "file_*" -follow -type f -print0 | xargs -0 zcat | agrep -dEOE 'grep'

答案 3 :(得分:8)

您还可以通过管道传输到while循环,该循环可以对find找到的文件执行多项操作。所以这里有一个用于查找jar档案中的给定java类文件的文件夹,其中包含大量jar个文件

find /usr/lib/eclipse/plugins -type f -name \*.jar | while read jar; do echo $jar; jar tf $jar | fgrep IObservableList ; done

关键点是while循环包含引用以分号分隔的传入文件名的多个命令,这些命令可以包含管道。因此,在该示例中,我回显匹配文件的名称,然后列出给定类名的归档过滤中的内容。输出如下:

/usr/lib/eclipse/plugins/org.eclipse.core.contenttype.source_3.4.1.R35x_v20090826-0451.jar /usr/lib/eclipse/plugins/org.eclipse.core.databinding.observable_1.2.0.M20090902-0800.jar 组织/蚀/核心/数据绑定/可观察的/列表/的创建IObservableList 的.class /usr/lib/eclipse/plugins/org.eclipse.search.source_3.5.1.r351_v20090708-0800.jar /usr/lib/eclipse/plugins/org.eclipse.jdt.apt.core.source_3.3.202.R35x_v20091130-2300.jar /usr/lib/eclipse/plugins/org.eclipse.cvs.source_1.0.400.v201002111343.jar /usr/lib/eclipse/plugins/org.eclipse.help.appserver_3.1.400.v20090429_1800.jar

在我的bash shell(xubuntu10.04 / xfce)中,它确实使匹配的classname变为粗体,因为fgrep突出显示匹配的字符串;这样可以很容易地扫描搜索过的数百个jar文件列表,并轻松查看任何匹配项。

在Windows上,您可以使用以下方法执行相同的操作:

for /R %j in (*.jar) do @echo %j & @jar tf %j | findstr IObservableList

请注意,在Windows上,命令分隔符是'&'不是';'并且'@'抑制命令的回显以提供整齐的输出,就像上面的linux查找输出一样;虽然findstr不会使匹配的字符串变为粗体,但您必须在输出处看得更近才能看到匹配的类名。事实证明,windows'for'命令知道了很多技巧,例如循环遍历文本文件......

享受

答案 4 :(得分:0)

如果您正在寻找一种简单的替代方法,则可以使用循环来完成:

from datetime import datetime

startTime = datetime.strptime(startTime, '%Y-%m-%d %H:%M:%S')
EndTime = datetime.strptime(EndTime, '%Y-%m-%d %H:%M:%S')

out = []
for x in files:
    date = datetime.strptime(x.split('_', 1)[1].split(.)[0],, '%Y-%m-%d_%H:%M:%S')
    if date >= startTime and date <= EndTime:
        out.append(pd.read_csv(x))

df = pd.concat(out, ignore_index)

或更一般的和易于理解的形式:

for i in $(find -name 'file_*' -follow -type f);do zcat $i | agrep -dEOE 'grep');done

并在YOUR_EXEC_COMMAND_AND_PIPES中用$ i替换任何{}

答案 5 :(得分:0)

我发现运行字符串shell命令(sh -c)效果最佳,例如:

find -name 'file_*' -follow -type f -exec bash -c "zcat \"{}\" | agrep -dEOE 'grep'" \;