我正在尝试使用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;
答案 0 :(得分:2)
Spirit规则通过引用保存其组件,因此您必须确保引用的组件保留在范围内,并且在引用时不会被破坏。看一下代码
rule<Iterator, Foo(), Skipper> foo = qi::lit("Foo") >> '.' >> FooValues();
在使用规则之前,FooValues的实例超出了范围。你可以用这种方式修复它
FooValues values;
rule<Iterator, Foo(), Skipper> foo = qi::lit("Foo") >> '.' >> values;