如何检查Perl标量是否包含对某个子例程的引用?

时间:2012-11-11 16:59:58

标签: perl

换句话说,我如何检查coderef“相等”?

smartmatch运算符不起作用for obvious reasons(将其视为CODE->(ANY)),但我已将其包含在示例中,以显示我所追求的内容:

use strict;
use warnings;
use feature 'say';

sub pick_at_random {

    my %table = @_;
    return ( values %table )[ rand( keys %table ) ];
}

my %lookup = ( A => \&foo,
               B => \&bar,
               C => \&baz );

my $selected = pick_at_random( %lookup );

say $selected ~~ \&foo ? "Got 'foo'" :
    $selected ~~ \&bar ? "Got 'bar'" :
    $selected ~~ \&baz ? "Got 'baz'" :
                         "Got nadda" ;

1 个答案:

答案 0 :(得分:11)

您可以使用普通(数字)相等(==),就像所有引用一样:

Perl> $selected == \&foo


Perl> $selected == \&bar


Perl> $selected == \&baz
1

Live in action here

当引用被赋予重载==0+的东西(这对于coderefs来说不太可能)时,它会中断。在这种情况下,您需要比较Scalar::Util::refaddr($selected)

来自man perlref

  

将引用用作数字会生成一个整数,表示其在内存中的存储位置。要做的唯一有用的事情是比较两个          用数字表示他们是否指的是同一地点。

      if ($ref1 == $ref2) {  # cheap numeric compare of references
           print "refs 1 and 2 refer to the same thing\n";
       }