我想在我的无功二进制函数中使用$a
和$b
变量,就像在sort {$a <=> $b} (1, 2, 3)
中完成一样,但我无法弄清楚为什么代码会像
#!/usr/bin/env perl
use strict;
use warnings;
Foo::Bar(sub { $a + $b });
package Foo;
sub Bar {
my ($function) = @_;
for my $i (1, 2, 3) {
local ($a, $b) = ($i, $i);
print $function->() . "\n";
}
}
不起作用。而
#!/usr/bin/env perl
use strict;
use warnings;
Foo::Bar(sub { $_ });
package Foo;
sub Bar {
my ($function) = @_;
for my $i (1, 2, 3) {
local $_ = $i;
print $function->() . "\n";
}
}
工作正常。
我做错了什么?
答案 0 :(得分:14)
$a
和$b
是特殊的包变量。您从Foo::Bar
个包中调用了main
,因此您需要设置$main::a
和$main::b
才能使其正常运行。您可以使用caller
获取调用包的名称。这应该有效:
#!/usr/bin/env perl
use strict;
use warnings;
Foo::Bar(sub { $a + $b });
package Foo;
sub Bar {
my ($function) = @_;
my $pkg = caller;
for my $i (1, 2, 3) {
no strict 'refs';
local *{ $pkg . '::a' } = \$i;
local *{ $pkg . '::b' } = \$i;
print $function->() . "\n";
}
}
答案 1 :(得分:0)
如果有人有兴趣,请复制并粘贴List::MoreUtils::PP v.0.428(截至2017年12月):
# begin copyrighted content
sub reduce_u(&@)
{
my $code = shift;
# Localise $a, $b
my ($caller_a, $caller_b) = do
{
my $pkg = caller();
no strict 'refs';
\*{$pkg . '::a'}, \*{$pkg . '::b'};
};
local (*$caller_a, *$caller_b);
*$caller_a = \();
for (0 .. $#_)
{
*$caller_b = \$_[$_];
*$caller_a = \($code->());
}
${*$caller_a};
}
# end copyrighted content
仅在受no strict 'refs';
影响的区域中与上述示例不同。如果没有严格的话,可能无法完成。