2

我想在 perl 中运行一个外部命令并过滤一些行。我不知道如何过滤去 stderr 的行。我现在有以下代码:

#!/usr/bin/env perl

use File::Spec;
#open STDERR, '>', File::Spec->devnull() or die "could not open STDERR: $!\n";

open(FILEHANDLE, '-|', 'Mycmd') or die "Cannot fork: $!\n";
open(STDERR, ">&FILEHANDLE");

while(defined(my $line = <FILEHANDLE>)) {
  chomp($line);
  if( $line =~ m/text1/ or
    $line =~ m/text2/ or
    $line =~ m/text3/
  ) {
    # do nothing
  }
  else {
    print "$line\n";
  }
}
close FILEHANDLE or die "child error: $!\n";

线

open(STDERR, ">&FILEHANDLE");

是我尝试重定向 stderr 以便能够使用 stdout 处理它但它不起作用的地方。

该解决方案必须在 Windows 中工作。

4

1 回答 1

3

参数中的 shell 重定向open可以在这里提供帮助:

open(FILEHANDLE, 'Mycmd 2>&1 |') or die "Cannot fork: $!\n";

现在FILEHANDLE将看到标准输出和标准错误的每一行Mycmd

要使用多参数open 重定向输出,您必须更加慎重。说Mycmd

#! /usr/bin/env perl
print "standard output\n";
warn  "standard error\n";

打开"-|"只给我们标准输出,所以如果我们运行

#! /usr/bin/env perl

use strict;
use warnings;

use 5.10.0;

my $pid = open my $fh, "-|", "Mycmd" // die "$0: fork: $!";

while (defined(my $line = <$fh>)) {
  chomp $line;
  print "got [$line]\n";
}

输出是

标准误
得到[标准输出]

请注意,标准输出Mycmd通过了驱动程序,而不是其标准错误。要同时获得两者,您必须模仿 shell 的重定向。

#! /usr/bin/env perl

use strict;
use warnings;

use 5.10.0;

my $pid = open my $fh, "-|" // die "$0: fork: $!";

if ($pid == 0) {
  open STDERR, ">&STDOUT" or die "$0: dup: $!";
  exec "Mycmd"            or die "$0: exec: $!";
}

while (defined(my $line = <$fh>)) {
  chomp $line;
  print "got [$line]\n";
}

现在输出是

得到[标准错误]
得到[标准输出]
于 2012-05-14T14:24:07.723 回答