在Marpa::R2的cpan页面中,我理解BNF(Backus Naur Form(,但我对操作回调非常着迷。
在下面的示例中,我了解到左右两个成员被传递给do_multiply
。我对此没有意见。问题是,我找不到任何关于这些论点的文件?
my $dsl = <<'END_OF_DSL';
:default ::= action => [name,values]
lexeme default = latm => 1
Calculator ::= Expression action => ::first
...
Term '*' Factor action => do_multiply
...
END_OF_DSL
my $grammar = Marpa::R2::Scanless::G->new( { source => $dsl } );
sub do_multiply { $_[1] * $_[2] }
什么是$_[0]
甚至$_[3]
?这些记录在哪里?即使在marpa官方网站上,我也看不到任何文档。
在另一个例子中,这里是chobora的答案,我们看到pair
指的是$_[2]
和$_[3]
:
BNF:片段
Hash ::= '(' Pairs ')' action => hash
Pairs ::= Pair+ action => pairs
Pair ::= '(' Key Value ')' action => pair
Key ::= String
核心代码:
$recce->read($input);
print Dumper $recce->value;
sub hash { $_[2] }
sub pairs { shift; +{ map @$_, @_ } }
sub pair { [ @_[2, 3] ] } # What is 2 and 3?
sub itself { $_[1] }
Marpe::R2文档说明如下:
sub My_Actions::do_add { my ( undef, $t1, undef, $t2 ) = @_; return $t1 + $t2; }
Perl语义闭包是回调。它们被称为每个节点在解析树中进行评估。
每个Perl语义闭包都用一个或多个参数调用。这个值操作的第一个参数总是每个解析树对象,回调可以将其用作草稿。在本例中不使用每个解析树对象。剩下的参数将是节点的"子级"的值——换句话说,计算的值对于其每个RHS符号,按顺序。如果操作针对空规则,每个解析树对象将是它唯一的参数。
每个值操作都应返回一个值。除了一个例外,该值作为参数传递给父节点。例外情况是开始规则的值。开始规则的返回值成为解析结果。
因此,这些函数被传递一个对象作为第一个参数。这是$_[0]
。
对于您的示例,文本似乎意味着以下参数$_[1]
和$_[2]
只是用于计算内容的值。
添加一些调试输出给了我以下内容:
use Data::Printer;
sub My_Actions::do_add {
my ( undef, $t1, undef, $t2 ) = @_;
say 'do_add';
p @_;
return $t1 + $t2;
}
sub My_Actions::do_multiply {
my ( undef, $t1, undef, $t2 ) = @_;
say 'do_multiply';
p @_;
return $t1 * $t2;
}
__END__
do_multiply
[
[0] {},
[1] 42,
[2] "*",
[3] 1
]
do_add
[
[0] {},
[1] 42,
[2] "+",
[3] 7
]