使用boost :: spirit :: qi :: symbols

时间:2016-01-15 14:46:42

标签: c++ boost

我正在尝试使用boost::spirit::qi::symbols来解析枚举值。我的代码如下:

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

using namespace std;
namespace qi = boost::spirit::qi;

enum class Foo { None, Bar = 42, Baz = 43 };

struct FooValues: qi::symbols<char, Foo>
{
    FooValues() { add("Bar", Foo::Bar)("Baz", Foo::Baz); }
};

int main()
{
    typedef std::string::iterator Iterator;
    typedef qi::space_type Skipper;
    using qi::rule;

    rule<Iterator, Foo(), Skipper> foo = qi::lit("Foo") >> '.' >> FooValues();
    rule<Iterator, Foo(), Skipper> root = foo | ('\"' >> foo >> '\"');

    std::string input = "Foo.Bar";
    Foo output;
    if (qi::phrase_parse(input.begin(), input.end(), root, Skipper(), output))
        cout << "output: " << static_cast<int>(output) << endl;
    else
        cout << "failed" << endl;
}

然而,它无法解析输入。如果我用以下方法替换解析枚举值的规则就可以了。

rule<Iterator, Foo(), Skipper> foo = qi::lit("Foo") >> '.' >>
  ((qi::lit("Bar") >> qi::attr(Foo::Bar)) | (qi::lit("Baz") >> qi::attr(Foo::Baz)));

我做错了什么?我更喜欢使用qi::symbols解析器,因为它允许我在循环中添加值而不是手动将它们写入规则。

FooValues()临时替换为本地变量,如下所示,但如果不需要,我宁愿不创建额外的变量。这可能吗?

FooValues fooValues;
rule<Iterator, Foo(), Skipper> foo = qi::lit("Foo") >> '.' >> fooValues;

1 个答案:

答案 0 :(得分:2)

Spirit规则通过引用保存其组件,因此您必须确保引用的组件保留在范围内,并且在引用时不会被破坏。看一下代码

rule<Iterator, Foo(), Skipper> foo = qi::lit("Foo") >> '.' >> FooValues(); 

在使用规则之前,FooValues的实例超出了范围。你可以用这种方式修复它

FooValues values;
rule<Iterator, Foo(), Skipper> foo = qi::lit("Foo") >> '.' >> values;