在我之前的问题中,有人建议我的boost::spirit::x3
解析器的性能可以通过解析为boost::string_view
使用raw
指令来提高。
但是,我很难编译它。这是我发现的:
之前
x3
,必须专门assign_to_attribute_from_iterators
(参见例如this SO answer)来处理raw
指令。x3
现在改用move_to
free 函数(参见例如这个 SO answer)。
因此,我添加了一个move_to
重载,如果我解析来自char*
:
#include <iostream>
#include <string>
#include <boost/utility/string_view.hpp>
namespace boost {
namespace spirit { namespace x3 { namespace traits {
template <typename It>
void move_to(It b, It e, boost::string_view& v)
{
v = boost::string_view(b, std::size_t(std::distance(b,e)));
}
} } }
} // namespace boost
#include <boost/spirit/home/x3.hpp>
namespace parser
{
namespace x3 = boost::spirit::x3;
using x3::char_;
using x3::raw;
const auto str = raw[ +~char_('_')] >> '_';
}
int main()
{
std::string input = "hello world_";
boost::string_view str;
parse(input.data(), input.data()+input.size(), parser::str, str);
std::cout << str;
}
但是,它不会编译:
1)如果我使用解析std::string::const_iterator
parse(input.cbegin(), input.cend(), parser::str, str);
的构造函数boost::string_view
需要 aconst char*
或 a std::string&
。
main.cpp:12:16: error: no matching function for call to 'boost::basic_string_view<char, std::char_traits<char> >::basic_string_view(__gnu_cxx::__normal_iterator<const char*, std::__cxx11::basic_string<char> >&, std::size_t)'
v = boost::string_view(b, std::size_t(std::distance(b,e)));
^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
如何实例化boost::string_view
from std::string::const_iterator
?
2) Ifboost/spirit/home/x3.hpp
包含在move_to
重载之前
为什么没有选择我的重载?它不是比中定义的任何一个更好的重载boost/spirit/home/x3/support/traits/move_to.hpp
吗?无论包含顺序如何,我如何确保选择我的超载?