使用谓词和替代方法,boost :: spirit :: multi_pass崩溃

时间:2016-12-30 22:09:35

标签: c++ boost boost-spirit boost-spirit-qi

运行以下代码会导致崩溃。为什么呢?

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

using namespace boost::spirit;
typedef multi_pass<
            std::string::const_iterator,
            iterator_policies::default_policy<
                iterator_policies::first_owner,
                iterator_policies::no_check,
                iterator_policies::buffering_input_iterator,
                iterator_policies::split_std_deque>>
        string_mp_iterator;

int main() {
    std::string input = "234";
    string_mp_iterator input_begin(input.begin()),
            input_end((string_mp_iterator()));
    qi::rule<string_mp_iterator, boost::variant<int, double>()> r =
            &qi::lit('1') >> qi::int_ | qi::double_;
    qi::parse(input_begin, input_end, r);
    return 0;
}

为了重现崩溃,我似乎需要同时使用谓词和后续替代,使用multi_pass迭代器,并且输入不满足谓词。

我觉得我在某种程度上错误地使用multi_pass,但我不知道问题到底是什么。

2 个答案:

答案 0 :(得分:2)

似乎你不能使用multi_pass迭代器包装std :: string,至少不能用iterator_policies::buffering_input_iterator std :: string有一个基于end的指针,而不是null。这就是迭代器出现不兼容的原因。如果您要解析std::string,请直接使用迭代器,因为它们符合multi_pass的要求。如果您打算更改为流(code sorta from here)

typedef std::istreambuf_iterator<char> base_iterator_type;
typedef boost::spirit::multi_pass<base_iterator_type> forward_iterator_type;

main( )
{
    std::istringstream input( "234" );

    base_iterator_type in_begin(input);
    base_iterator_type in_end;
    forward_iterator_type fwd_begin = boost::spirit::make_default_multi_pass(in_begin);
    forward_iterator_type fwd_end   = boost::spirit::make_default_multi_pass(in_end);

    qi::rule<forward_iterator_type, boost::variant<int, double>()> r =
        &qi::lit('1') >> qi::int_ | qi::double_;
    qi::parse(fwd_begin, fwd_end, r);
    return 0;
}

答案 1 :(得分:2)

只需修复结束迭代器的初始化程序。

string_mp_iterator input_end(input.end());

由于它不是输入迭代器,因此不能合法使用默认构造的迭代器。

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

using namespace boost::spirit;
typedef multi_pass<
    std::string::const_iterator,
    iterator_policies::default_policy<
        iterator_policies::first_owner, iterator_policies::no_check,
        iterator_policies::buffering_input_iterator,
        iterator_policies::split_std_deque>>
    string_mp_iterator;

int main() {
    std::string input = "234";
    string_mp_iterator input_begin(input.begin()),
                       input_end(input.end());
    qi::rule<string_mp_iterator, boost::variant<int, double>()> r = &qi::lit('1') >> qi::int_ | qi::double_;
    qi::parse(input_begin, input_end, r);
}