假设我们有:
sub test {
print "testing\n";
}
如果有一种情况我想让它打印到stderr而不是stdout,有没有办法可以调用子程序来执行此操作?或者我可以将输出捕获到变量然后使用警告?我对perl很新。
答案 0 :(得分:12)
是的。 print
将其输出发送到“选定”文件句柄,通常为STDOUT
。但是Perl提供了select
函数供您更改。
select(STDERR);
&test; # send output to STDERR
select(STDOUT); # restore default output handle
select
函数返回先前选择的文件句柄,因此您可以捕获它并在以后恢复它。
my $orig_select = select(STDERR);
&test;
select($orig_select);
答案 1 :(得分:9)
Perl通过local()的动态范围经常被使用,但这对我来说是一个很好的应用:
test(); # to stdout
{
open(local *STDOUT, ">&STDERR") or die "dup out to err: $!";
test(); # to stderr, locally calling it "STDOUT"
}
test(); # to stdout again
对上面某个块中test()
的调用 - 以及test()
本身可能调用的任何内容 - 都会将STDOUT动态地限定为您的STDERR副本。当控制离开块时,即使通过die()
,STDOUT也将恢复为块之前的任何内容
广义:
sub out2err(&) {
my $user_block = shift;
open(local *STDOUT, ">&STDERR") or die $!;
$user_block->();
}
test(); # to stdout
out2err { test() }; # to stderr
test(); # to stdout
答案 2 :(得分:3)
同时,您还可以“将子程序的打印输出捕获到变量。”
只需将标量参考传递给open
:
#! /usr/bin/env perl
use common::sense;
use autodie;
sub tostring (&) {
my $s;
open local *STDOUT, '>', \$s;
shift->();
$s
}
sub fake {
say 'lalala';
say 'more stuff';
say 1 + 1, ' = 2';
say for @_;
}
for (tostring { fake(1, 2, 3) }) {
s/\n/\\n/g;
say "Captured as string: >>>$_<<<";
}
输出:
Captured as string: >>>lalala\nmore stuff\n2 = 2\n1\n2\n3\n<<<
答案 3 :(得分:0)
这对我有用
local *STDOUT;
open(STDOUT, ">", \$Result);
&test();
print $Result;