Perl - 将代码块作为括号内的参数传递

时间:2011-03-07 14:19:16

标签: perl perl5 perlsyn

是否可以使用“括号”语法将代码块传递给子?

即。当我写作

List::MoreUtils::any { defined ($_) } (undef, undef, 1);

它有效。但是当我尝试添加括号时

List::MoreUtils::any ( { defined ($_) } , (undef, undef, 1) );

这被解释为匿名哈希,给出错误消息。逃避或使用eval都没有帮助。

所有大惊小怪的想法是,如果调用是表达式的一部分,即

if (first_index { defined (${$_})} $jms_positions > $jms_positionals_seen )

跟随参数的某个运算符可能会在调用之前执行,从而产生不希望的结果。

3 个答案:

答案 0 :(得分:15)

使用语法

声明匿名子例程
sub { say "The sub with no name!" };

Perl的原型系统允许一个特殊的例外,其中代码块是第一个参数,在这种情况下,您可以不使用前导sub并且只是传递块,类似于Perl内置。但这只适用于在无括号的样式中调用它。使用parens会导致解析器认为您要传递hash-ref。

所以你可以说

List::MoreUtils::any( sub { defined }, undef, undef, 1 );

如果你坚持使用parens。

答案 1 :(得分:5)

不,仅适用于内置组件。

$ perl -ce'map({ "..." } @a)'
-e syntax OK

$ perl -ce'grep({ "..." } @a)'
-e syntax OK

$ perl -ce'sub f(&@); f { "..." } @a'
-e syntax OK

$ perl -ce'sub f(&@); f({ "..." } @a)'
Array found where operator expected at -e line 1, near "} "
        (Missing operator before  ?)
syntax error at -e line 1, near "} @a"
-e had compilation errors.

在整个通话中添加parens通常是一种合适的解决方法。

( any { defined } (undef, undef, 1) )

答案 2 :(得分:3)

只需插入sub

List::MoreUtils::any ( sub { defined ($_) } , (undef, undef, 1) );