Antlr 4 C++ 目标语法规则返回 lambda,规则引用错误时缺少属性访问

Antlr 4 C++ target grammar rule returning lambda, missing attribute access on rule reference error

我是 C++ 和 Antlr 的新手,请原谅我的无知。

我需要根据多个规则导出数百万个值。

Eg rule1:-  Value = ob.field1 * ob.field2 //the user defines the rule
Eg rule2:-  Value = 4* ob.field4 < 3* ob.field1 ? 5 : 0b.field6 

所以,我只需要解析一次规则并生成仿函数(或 lambda),这样我就可以将它们保存在地图中并随时调用它们。 Ony ob 实例每次都不同。

这是我想出的简单示例,k 是我作为此示例的参数传递的双精度值,稍后它将是一个对象。

grammar calculator;


start: expr EOF;

expr returns [std::function<double(double)> exprEval]
    : left=expr op=('+'|'-') right=expr {$exprEval= [](double k)->double { return $left.exprEval(k) + $right.exprEval(k); }; }
    | left=expr op=('*'|'/') right=expr {$exprEval= [](double k)->double { std::cout<<2*k<<std::endl;  return -2*k; }; }
    | '(' expr ')'                      {$exprEval= [](double k)->double { std::cout<<k<<std::endl;    return -1*k; }; }
    | numb                              {$exprEval= [](double k)->double { std::cout<<-1*k<<std::endl; return k;    }; }
    ;

numb 
    :DOUBLE
    |INT 
    ;

INT 
    : [0-9]+
    ;

DOUBLE
    : [0-9]+'.'[0-9]+
    ;

WS
   : [ \r\n\t] + -> channel (HIDDEN)
   ;

它产生以下错误。我想我引用他们是错误的。

error(67): calculator.g4:6:152: missing attribute access on rule reference left in $left
error(67): calculator.g4:6:172: missing attribute access on rule reference right in $right

以下也不起作用。

$left.ctx.exprEval(k) //compilation error :  in lambda, localctx is not captured.
ctx.$left.exprEval(k) //compilation error : ctx was not declared in this scope

如何从 lambda 内部访问 "left" 和 "right" 表达式上下文? 或者这不是最好的方法吗?有没有更好的方法?

我认为每次都解析规则不是一个好主意,因为有数百万条记录。

您可能可以通过调整 lambda 中的捕获来逃脱,但我强烈建议您改变您的方法。不要在语法中编写所有代码,而是创建一个侦听器(如果需要评估表达式,则创建一个访问者)并在那里实现所有代码。更容易维护,您可以避免这样的麻烦。