c++ - 使用 Boost Spirit 的指数运算符表达式语法

标签 c++ boost-spirit context-free-grammar boost-spirit-qi associativity

我想将求幂运算符添加到 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 star 是不正确的。将语法转换为 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

最佳答案

我认为你可以使用正确的递归来得到你想要的:

factor= expo >> -('^' >> factor [&do_power]);

我不确定所需的评估顺序;你可能想要类似的东西

factor= expo [&do_power] >> -('^' >> factor);

相反。

这是一个简单的测试程序,展示了它如何处理 2^(6/2)^4+1:

Edit See it LIVE on Coliru

Type an expression...or [q or Q] to quit
2^(6/2)^4+1

push 2
push 6
push 2
divide
push 4
exp
exp
push 1
add
-------------------------
Parsing succeeded

完整代码

#define BOOST_SPIRIT_NO_PREDEFINED_TERMINALS
#define BOOST_SPIRIT_DEBUG

#include <boost/spirit/include/qi.hpp>

namespace client
{
    namespace qi = boost::spirit::qi;
    namespace ascii = boost::spirit::ascii;

    ///////////////////////////////////////////////////////////////////////////////
    //  Semantic actions
    ////////////////////////////////////////////////////////1///////////////////////
    namespace
    {
        void do_int(int n)  { std::cout << "push " << n << std::endl; }
        void do_add()       { std::cout << "add\n"; }
        void do_subt()      { std::cout << "subtract\n"; }
        void do_mult()      { std::cout << "mult\n"; }
        void do_div()       { std::cout << "divide\n"; }
        void do_power()     { std::cout << "exp\n"; }
        void do_neg()       { std::cout << "negate\n"; }
    }

    ///////////////////////////////////////////////////////////////////////////////
    //  Our calculator grammar
    ///////////////////////////////////////////////////////////////////////////////
    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 >> -('^' >> factor [&do_power]);

            expo = uint_                        [&do_int]
                |  '(' >> expression >> ')'
                |  ('-' >> expo[&do_neg])
                |  ('+' >> expo)
            ;

            BOOST_SPIRIT_DEBUG_NODES((expression)(term)(factor)(expo));
        }
      private:
        qi::rule<Iterator, ascii::space_type> expression, term, factor, expo;
    };
}

///////////////////////////////////////////////////////////////////////////////
//  Main program
///////////////////////////////////////////////////////////////////////////////
int
main()
{
    std::cout << "/////////////////////////////////////////////////////////\n\n";
    std::cout << "Expression parser...\n\n";
    std::cout << "/////////////////////////////////////////////////////////\n\n";
    std::cout << "Type an expression...or [q or Q] to quit\n\n";

    typedef std::string::const_iterator iterator_type;
    typedef client::calculator<iterator_type> calculator;

    boost::spirit::ascii::space_type space; // Our skipper
    calculator calc; // Our grammar

    std::string str;
    while (std::getline(std::cin, str))
    {
        if (str.empty() || str[0] == 'q' || str[0] == 'Q')
            break;

        std::string::const_iterator iter = str.begin();
        std::string::const_iterator end = str.end();
        bool r = phrase_parse(iter, end, calc, space);

        if (r && iter == end)
        {
            std::cout << "-------------------------\n";
            std::cout << "Parsing succeeded\n";
            std::cout << "-------------------------\n";
        }
        else
        {
            std::string rest(iter, end);
            std::cout << "-------------------------\n";
            std::cout << "Parsing failed\n";
            std::cout << "stopped at: \" " << rest << "\"\n";
            std::cout << "-------------------------\n";
        }
    }

    std::cout << "Bye... :-) \n\n";
    return 0;
}

关于c++ - 使用 Boost Spirit 的指数运算符表达式语法,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/17619113/

相关文章:

c++ - 如何使用 Eigen 设置 Levenberg-Marquardt 阻尼

c++ - 在 Rcpp Makevars 中设置 CXXFLAGS

c++ - boost spirit语法的不一致行为

c++ - 如何获取 boost.spirit 数字解析器匹配的子字符串?

python - NLTK 正则表达式和 CFG

algorithm - 有没有一种通用的方法可以将明确的上下文无关文法转换为 LALR(1) 文法?

c++ - Gtkmm 3.0 条目 get_text 始终返回相同的初始文本

c++ - 部分文本上的并发 mmap()

c++ - boost spirit 解析器前瞻解析

java - 在 Java 中验证给定上下文无关语法的字符串