如何在 Perl 中捕获输出和退出代码时执行外部脚本?
Posted
技术标签:
【中文标题】如何在 Perl 中捕获输出和退出代码时执行外部脚本?【英文标题】:How do I execute an external script while capturing both output and exit code in Perl? 【发布时间】:2010-02-05 09:17:42 【问题描述】:我正在尝试从 Perl 脚本中检查是否存在 SVN 标记。所以我尝试调用svn info $url
,读取退出代码并抑制标准输出和标准错误流。然而,我很难优雅地做到这一点(可能有更好的方法向 SVN 询问标签,但这不是重点):
my $output = `svn info $url/tags/$tag`;
这会在将其放入$output
时抑制输出。退出代码丢失。
my $output = `svn info $url/tags/$tag 2>&1`;
这会抑制 STDERR 和 STDOUT 并将它们都放入$output
。退出代码再次丢失。
my $exitcode = system("svn", "info", "$url/tags/$tag");
这会捕获退出代码,但用户可以看到实际的输出和错误流。
open( STDERR, q>, "/dev/null" );
open my $fh, q>, "/dev/null";
select($fh);
if (system("svn", "info", "$url/tags/$tag") != 0)
select(STDOUT);
print ("Tag doesn't exist!");
do_something_with_exit();
select(STDOUT);
print "Exit code: $exitcode";
这会杀死 STDOUT 和 STDERR 并捕获退出代码,但这很丑,因为我必须记住将 STDOUT 切换回原始代码。
那么,有没有更优雅的解决方案?
【问题讨论】:
【参考方案1】:尝试使用$?
.e.g.
my $output = `svn info $url/tags/$tag`;
my $extcode = $?>>8;
【讨论】:
退出码是$?
中的高字节,所以需要$?>> 8
。【参考方案2】:
当您尝试使用IPC::System::Simple 时会发生什么?该模块处理这类问题的大部分细节:
use IPC::System::Simple qw(capturex $EXITVAL);
my $output = capturex( "some_command", @args );
my $exit = $EXITVAL;
【讨论】:
【参考方案3】: my $output = `svn info $url/tags/$tag 2>&1`;
这会抑制 STDERR 和 STDOUT 并将它们都放入 $output 中。退出代码再次丢失
您确定退出代码丢失了吗?当我尝试这个时,我会在$?
中获得退出代码。
【讨论】:
嗯,退出码是$?
中的高字节,所以你需要$? >> 8
。【参考方案4】:
IPC::Run3 模块对输入和输出提供了非常细粒度的控制。
use IPC::Run3;
run3 \@cmd, \$in, \$out, \$err;
您可以将相同的变量传递给\$out
和\$err
,它会按照您的预期执行,将两个流结合起来。输入是不必要的,因此您可以传递undef
(“从父进程继承”)或\undef
(“关闭的文件句柄”。)
IPC::Run3::run3()
根据退出代码返回 true 或 false,并将子进程的实际退出代码保留在 $?
中,按照“perlvar”。
在你的情况下,你会运行
use IPC::Run3
my @cmd = ('svn', 'info', "$url/tags/$tag");
my $out;
my $rv = run3(\@cmd, \undef, \$out, \$out);
if ($rv)
# process $out
else
die "error: $@";
【讨论】:
run3 的返回值只处理文件句柄的处理。对进程的退出代码没有意义。以上是关于如何在 Perl 中捕获输出和退出代码时执行外部脚本?的主要内容,如果未能解决你的问题,请参考以下文章
通过perl的open2执行时,telnet命令输出未完全打印