【发布时间】:2013-07-12 16:01:50
【问题描述】:
我想将求幂运算符添加到expression grammar provided in the Boost spirit samples。
BNF 语法如下:(请参阅此答案,例如:"Unambiguous grammar for exponentiation operation")
E -> E + T | E - T | T
T -> T * F | T / F | X
X -> X ^ Y | Y
Y -> i | (E)
我这样翻译成 Boost Spirit:
template <typename Iterator>
struct calculator : qi::grammar<Iterator, ascii::space_type>
{
calculator() : calculator::base_type(expression)
{
qi::uint_type uint_;
expression =
term
>> *( ('+' >> term [&do_add])
| ('-' >> term [&do_subt])
)
;
term =
factor
>> *( ( '*' >> factor [&do_mult])
| ('x' >> factor [&do_mult])
| ('/' >> factor [&do_div])
);
factor= expo >> *( '^' >> expo [&do_power]);
expo =
uint_ [&do_int]
| '(' >> expression >> ')'
| ('-' >> expo[&do_neg])
| ('+' >> expo)
;
}
qi::rule<Iterator, ascii::space_type> expression, term, factor, expo;
};
问题在于这种情况下的 ^ 运算符是左关联的,即 2 ^ 3 ^ 4 被错误地解析为 (2 ^ 3) ^ 4 而不是 2^ (3 ^ 4)。
如何重写语法以使^ 变为右结合?显然我在factor 的定义中使用的Kleene 星是不正确的。将语法转换为 Spirit 代码的方法是什么?似乎有一种方法可以从左因子语法转到 Spirit 实现,但我无法立即看到。
以更正式的方式,Spirit 代码如下所示(在我尝试添加指数之前):
E = T ( +T | -T ) *
T = F ( xF | /F ) *
F = int | ( E ) | +F | -F
而左因子语法是
E = T E'
E' = +T E' | -T E' | epsilon
T = F T'
T' = *F T' | /F T' | epsilon
F = ( E ) | int | +F | -F
【问题讨论】:
标签: c++ boost-spirit context-free-grammar boost-spirit-qi associativity