在Perl中分配和测试定义时,了解优先级

在尝试分配变量并在Perl中的一个操作中测试它的定义时,例如在if的条件下会很有用,我写它似乎很自然:
if ( defined my $thing = $object->get_thing ) {
    $thing->do_something;
}

据我所知,定义的优先级为rightward list operator,低于赋值的优先级,因此我希望上面的代码等同于:

if ( defined ( my $thing = $object->get_thing ) ) {
    $thing->do_something;
}

虽然后者,带括号的代码确实有效,但前者会产生以下致命错误:“无法在标量赋值中修改已定义的运算符”.

添加括号并不是什么大不了的事,但我很想理解为什么第一个版本不起作用,例如什么样的“事物”定义是什么,它的优先级是什么?

解决方法

命名运算符分为一元运算符(总是只运算一个操作数的运算符)和列表运算符(其他所有)[1].

已定义且我的[2]是一元运算符,其优先级高于其他命名运算符.

对于subs来说也是如此,所以我将用它们来演示.

$perl -MO=Deparse,-p -e'sub f :lvalue {}  sub g :lvalue {}  f g $x = 123;'
sub f : lvalue { }
sub g : lvalue { }
f(g(($x = 123)));
-e Syntax OK

$perl -MO=Deparse,-p -e'sub f($) :lvalue {}  sub g($) :lvalue {}  f g $x = 123;'
sub f ($) : lvalue { }
sub g ($) : lvalue { }
(f(g($x)) = 123);
-e Syntax OK

但是,当然,定义的不是左值函数,因此在赋值的LHS上找到它会导致错误.

>和,或者,不是,xor,lt,le,gt,ge,eq,ne和cmp不被视为命名运算符.>我很不寻常.除了具有编译时和运行时效果之外,其语法还取决于是否在其参数周围使用了parens.没有parens,它是一个一元的操作符.有了parens,它是一个列表操作符.

相关文章

1. 如何去重 #!/usr/bin/perl use strict; my %hash; while(...
最近写了一个perl脚本,实现的功能是将表格中其中两列的数据...
表的数据字典格式如下:如果手动写MySQL建表语句,确认麻烦,...
巡检类工作经常会出具日报,最近在原有日报的基础上又新增了...
在实际生产环境中,常常需要从后台日志中截取报文,报文的形...
最近写的一个perl程序,通过关键词匹配统计其出现的频率,让...