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