运行以下代码会导致崩溃。为什么呢?
#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
,但我不知道问题到底是什么。
答案 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);
}