我想将求幂运算符添加到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 精神:
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)
.
我如何重写语法以使其^
成为右联想?显然,我在定义中使用的 Kleene 星factor
是不正确的。将语法转换为 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