0

我的问题如下。我有一个 ast 节点,定义如下:

struct foo_node{
    std::vector<std::string> value;
}

我有一个像这样的解析器来解析结构,它工作正常:

typedef x3::rule<struct foo_node_class, foo_node> foo_node_type;
const foo_node_type foo_node = "foo_node";
auto const foo_node_def = "(" >> +x3::string("bar") >> ")";

现在我想实现解析器也解析"bar",不带括号,但前提是它只有一个条。我试着这样做:

    auto const foo_node_def = x3::string("bar") 
                             | "(" > +x3::string("bar") > ")";

但这给了我一个编译时错误,因为x3::string("bar")返回一个字符串而不是一个std::vector<std::string>. 我的问题是,我怎样才能实现x3::string("bar")解析器(以及所有其他返回字符串的解析器)解析为向量?

4

1 回答 1

2

解析单个元素并将其公开为单个元素容器属性的方法是x3::repeat(1) [ p ]

Live On Coliru

#include <boost/fusion/adapted/struct.hpp>
#include <boost/spirit/home/x3.hpp>
#include <iostream>

namespace x3 = boost::spirit::x3;

struct foo_node {
    std::vector<std::string> value;
};

BOOST_FUSION_ADAPT_STRUCT(foo_node, value)

namespace rules {
    auto const bar 
        = x3::string("bar");

    auto const foo_node
        = '(' >> +bar >> ')'
        | x3::repeat(1) [ +bar ]
        ;
}

int main() {
    for (std::string const input : {
            "bar",
            "(bar)",
            "(barbar)",
            })
    {
        auto f = input.begin(), l = input.end();

        foo_node data;
        bool ok = x3::parse(f, l, rules::foo_node, data);

        if (ok) {
            std::cout << "Parse success: " << data.value.size() << " elements\n";
        } else {
            std::cout << "Parse failed\n";
        }

        if (f != l)
            std::cout << "Remaining unparsed: '" << std::string(f,l) << "'\n";
    }
}

印刷

Parse success: 1 elements
Parse success: 1 elements
Parse success: 2 elements
于 2016-08-21T19:01:59.030 回答