在Perl中捕获信号的最优雅方法是什么?

时间:2011-05-28 06:48:31

标签: perl exception signals

我在Perl脚本中有两条线可以抛出__WARN__。如果第一个抛出,那么我想从函数返回而不是尝试继续。

我知道如何在两行之前设置处理程序,以便我可以报告错误等:

local $SIG{__WARN__} = sub {
  my $e = shift;
  # log the error etc.
  return;
};
# possibly warning-resulting line 1
# possibly warning-resulting line 2

但是这两条线都会发生这种情况。我宁愿只抓住第一个实例并从函数返回。但是该处理程序中的返回仅返回处理程序,而不是外部函数。

处理信号时有没有办法从函数返回?

4 个答案:

答案 0 :(得分:2)

将两行包装在单独的函数中,并让第一行返回一个状态,指示调用函数应该返回。单独的函数可以根据需要处理警告 - 可能使用相同的函数来执行相同的日志记录。

sub wrap_line_1
{
    local $SIG{__WARN__} = ...;
    ...do line 1...
    return ($warning_fired ? 1 : 0);
}

sub wrap_line_2
{
    local $SIG{__WARN__} = ...;
    ...do line 2...
    return;
}


...calling code...
wrap_line_1() and return;
wrap_line_2();
...

答案 1 :(得分:1)

不,你不能强迫来电者回来。 (好吧,我相信你可以,用正确的黑魔法XS咒语。但不要!)

这似乎是一种误入歧途的设计;警告不应该是致命的错误,当然不应该像这样使用。

答案 2 :(得分:0)

只需抛出异常并捕获并返回调用者。

让处理程序返回一个值并检查它以在eval之后的if语句中返回。

实际上你甚至不必从处理程序中返回一个值,因为它不是决定是否返回 - 你总是在捕获时返回。

具体做法是:

# We're in the caller now
eval{
    local $SIG{__WARN__} = sub {
        my $e = shift;
        # log the error etc.
        die MyWarnException->new("This is an exception.");
    };

    # Offending statements go here
    do_things();

};
if( $@ && $@->isa('MyWarnException')){
    return;
}

答案 3 :(得分:0)

你可以在你的处理程序中死掉()并在eval中捕获它,如下所示:

use 5.12.0;

local $SIG{__WARN__} = sub {
  my $e = shift;
  # log the error etc.
  die "$e";
};

func(undef);
func('honk');

sub func {
    my $foo = shift;

    eval {
       my $bar = "a $foo";
       say "$bar";
    };
    if ($@) {
       die $@ unless $@ =~ /^Use of uninitialized value/;
    }
    return;
}

但是,您可能希望直接检查警告是否可能发生并直接返回:

sub func {
  my $foo = shift;
  return unless defined($foo);
  ...; # use $foo fine
}