boost::spirit算术公式解析器编译失败
boost::spirit arithmetic formulas parser fails to compile
我正在尝试为算术表达式编写一个spirit解析器,它填充一个抽象语法树。如果我没有尝试填充AST,但在当前版本中失败(有一个24K错误),解析器就会编译。我使用clang++3.5.0和-std=c++11,并在Ubuntu 14.4上运行。
#include <string>
#include <vector>
#include <utility>
#include <boost/spirit/include/qi.hpp>
#include <boost/fusion/include/adapt_struct.hpp>
#include <boost/fusion/adapted.hpp>
#include <boost/fusion/include/adapted.hpp>
#include <boost/variant/variant.hpp>
#include <boost/variant/recursive_wrapper.hpp>
using std::string;
using std::vector;
using std::pair;
using boost::spirit::qi::grammar;
using boost::spirit::qi::space_type;
using boost::spirit::qi::rule;
struct Term; // forward dec
typedef boost::recursive_wrapper<Term> RWTerm;
typedef pair<char, RWTerm> OpAndRWTerm;
typedef pair<RWTerm, vector<OpAndRWTerm> > Expr;
typedef boost::variant<Expr, double> Factor;
typedef pair<char, Factor> OpAndFactor;
struct Term : public pair<Factor, vector<OpAndFactor> >{};
template<typename It>
struct formula_parser : grammar<It, Expr(), space_type> {
formula_parser() : formula_parser::base_type(expr_rule) {
using boost::spirit::qi::double_;
using boost::spirit::ascii::char_;
factor_rule %= double_ | parenthesis_rule;
parenthesis_rule %= '(' >> expr_rule >> ')';
op_and_factor_rule %= char_("/*") >> factor_rule;
term_rule %= factor_rule >> *op_and_factor_rule;
op_and_term_rule %= char_("+-") >> term_rule;
expr_rule %= term_rule >> *op_and_term_rule;
}
rule<It, OpAndRWTerm(), space_type> op_and_term_rule;
rule<It, Expr(), space_type> expr_rule;
rule<It, OpAndFactor(), space_type> op_and_factor_rule;
rule<It, RWTerm(), space_type> term_rule;
rule<It, Expr(), space_type> parenthesis_rule;
rule<It, Factor(), space_type> factor_rule;
};
int main() {
formula_parser<string::const_iterator> grammar;
}
我从错误消息中了解到,融合混淆了规则term_rule中的Types Factor和RWTerm。
我做错了什么?
如果我更改两件事,它会为我编译:
-
由于
Term
继承自std::pair
,因此Term
是新的类型。因此,您需要将BOOST_FUSION_ADAPT_STRUCT
应用于Term
,而不管是否已经在<boost/fusion/adapted/std_pair.hpp>
:中为std::pair
执行了此操作BOOST_FUSION_ADAPT_STRUCT( Term, (Factor, first) (std::vector<OpAndFactor>, second) )
或者,您可以使
Term
成为一个具有两个成员的独立结构,然后在其上应用BOOST_FUSION_ADAPT_STRUCT
:struct Term { Factor f; std::vector<OpAndFactor> o;}; BOOST_FUSION_ADAPT_STRUCT( Term, (Factor, f) (std::vector<OpAndFactor>, o) )
顺便说一句:你必须在这里完全限定
std::vector
,因为以下内容不会编译:using std::vector; BOOST_FUSION_ADAPT_STRUCT( Term, (Factor, f) (vector<OpAndFactor>, o) )
-
声明
term_rule
:时使用Term
而不是RWTerm
rule<It, Term(), space_type> term_rule;
完整代码:
#include <string>
#include <vector>
#include <utility>
#include <boost/spirit/include/qi.hpp>
#include <boost/fusion/include/adapt_struct.hpp>
#include <boost/fusion/adapted.hpp>
#include <boost/fusion/include/adapted.hpp>
#include <boost/variant/variant.hpp>
#include <boost/variant/recursive_wrapper.hpp>
using boost::spirit::qi::grammar;
using boost::spirit::qi::space_type;
using boost::spirit::qi::rule;
struct Term; // forward dec
typedef boost::recursive_wrapper<Term> RWTerm;
typedef std::pair<char, RWTerm> OpAndRWTerm;
typedef std::pair<RWTerm, std::vector<OpAndRWTerm> > Expr;
typedef boost::variant<Expr, double> Factor;
typedef std::pair<char, Factor> OpAndFactor;
struct Term : public std::pair<Factor, std::vector<OpAndFactor> >{};
BOOST_FUSION_ADAPT_STRUCT(
Term,
(Factor, first)
(std::vector<OpAndFactor>, second)
)
template<typename It>
struct formula_parser : grammar<It, Expr(), space_type> {
formula_parser() : formula_parser::base_type(expr_rule) {
using boost::spirit::qi::double_;
using boost::spirit::ascii::char_;
factor_rule %= double_ | parenthesis_rule;
parenthesis_rule %= '(' >> expr_rule >> ')';
op_and_factor_rule %= char_("/*") >> factor_rule;
term_rule %= factor_rule >> *op_and_factor_rule;
op_and_term_rule %= char_("+-") >> term_rule;
expr_rule %= term_rule >> *op_and_term_rule;
}
rule<It, OpAndRWTerm(), space_type> op_and_term_rule;
rule<It, Expr(), space_type> expr_rule;
rule<It, OpAndFactor(), space_type> op_and_factor_rule;
rule<It, Term(), space_type> term_rule;
rule<It, Expr(), space_type> parenthesis_rule;
rule<It, Factor(), space_type> factor_rule;
};
int main() {
formula_parser<std::string::const_iterator> grammar;
}
实例
相关文章:
- luaL_dofile在已知良好的字节码上失败,可以使用未编译的版本
- 包含模板文件的递归会导致编译失败
- 提升 1.64 单元测试编译失败
- 如何让谷歌测试正常运行。测试总是失败。(它不会编译)
- 运行时检查失败 #0 用于运行时重新编译
- C++模板的模板编译失败
- 使用已删除的函数进行编译失败,并显示 uclibc
- MinGW-w64 编译失败,nullptr_t
- 使用提升线程时编译失败
- 编译花絮 g++, clang++, 使用 libboost -- g++8 编译失败时 g++7 成功;
- 编译成功,使用 clang 5.0.1/6.0.0 ,在 5.0.2/6.0.1 时失败
- CMake + Qt,moc 编译失败,无法实现 QMetaObject 方法(编译器找不到基本 ui 对象的标头?
- 带有引用的std::tuple在clang中编译失败,但在gcc中编译失败
- bitbake-grpc交叉编译/配置失败,错误为c-ares::care引用文件/usr/lib/libcares.s
- 解决由于在哈希函数中使用了不完整的类型而导致的编译失败
- 将XCode升级到verison 11.1后,C++编译失败
- 继承的构造函数,在 clang++3.9 中编译,在 g++ 7 中失败
- C++:此代码可以编译,但引发运行时检查失败 #2 - 围绕变量周围的堆栈'num'已损坏。发生
- 添加类型名会导致程序编译失败
- 将 qi::lexeme 添加到灵气中的规则时编译失败