解析字符串对失败。 Bad spirit x3语法
Parsing pair of strings fails. Bad spirit x3 grammar
我想解析键值对,将字符串映射到字符串。因为我想在右手边支持 { ... }
块,所以我想出了一个简单的语法来开始
#include <boost/spirit/home/x3.hpp>
#include <boost/fusion/include/std_pair.hpp>
namespace grammar::map
{
using namespace boost::spirit::x3;
auto expression = rule<class expression, std::pair<std::string, std::string>>{"expression"};
auto lhs = *(~char_('='));
auto rest = *(~char_(';'));
auto block = '{' >> *expression >> '}';
auto expression_def = lhs >> '=' >> (block | rest) >> ';';
BOOST_SPIRIT_DEFINE(expression)
}
但是组合编译失败,除非我把expression的属性改成std::string
.
//! Transform a string into a key-value map of strings.
template <class M, class R>
requires InputRange<R>() && _ContainerLike<M>
&& Same<value_type_t<M>, std::pair<const std::string, std::string>>()
// && Assignable<std::pair<std::string, std::string>, value_type_t<M>>()
M parse(R&& range)
{
auto begin = rng::begin(range);
auto end = rng::end(range);
auto map = M{};
auto ret = x3::phrase_parse(begin, end, *grammar::map::expression, x3::space, map);
if (!ret)
throw std::runtime_error{"parse error"};
return map;
}
我收到错误
boost/spirit/home/x3/support/traits/move_to.hpp:62:18: error: cannot convert ‘std::remove_reference<std::pair<std::__cxx11::basic_string<char>, std::__cxx11::basic_string<char> >&>::type {aka std::pair<std::__cxx11::basic_string<char>, std::__cxx11::basic_string<char> >}’ to ‘char’ in assignment
dest = std::move(src);
位于
boost/spirit/home/x3/support/traits/move_to.hpp: In instantiation of ‘void boost::spirit::x3::traits::detail::move_to_plain(Source&&, Dest&, mpl_::false_) [with Source = std::pair<std::__cxx11::basic_string<char>, std::__cxx11::basic_string<char> >; Dest = char; mpl_::false_ = mpl_::bool_<false>]’:
如果尝试以下表达式,同样的事情会发生
auto pair = std::pair<std::string, std::string>{};
auto ret = x3::phrase_parse(begin, end, grammar::map::expression, x3::space, map);
几天前我就开始使用它,但不知道如何正确地进行操作。感谢您的帮助...:^)
使用 boost-1.60{-62} 和 gcc 6.1.1、6.2 以及较新的主干版本对其进行了测试。
你的问题是你定义了 expression
有一个 pair<string,string>
的属性,但是 lhs >> '=' >> (block|rest) >> ';'
的综合属性是 synt_attr=tuple<string,variant<vector<synt_attr>,string>>
基本上是 synt_attr=pair<string,variant<map_of_value_type_synt_attr,string>>
.因此,根据您想要的结果,您至少有两个选择:
将合成属性更改为pair<string,string>
。使用 x3::raw
指令 (running on WandBox):
非常容易
auto expression_def = lhs >> '=' >> (raw[block] | rest) >> ';';
更改您对 expression
的定义,使其具有与合成属性兼容的属性。这需要使用递归变体并使您需要访问已解析地图中的数据的方式复杂化,因为您需要创建访问者 (running on WandBox)。
//A value is either a `string` or a `map<string,Value>`
using Value = boost::make_recursive_variant<std::string,std::map<std::string,boost::recursive_variant_>>::type;
...
struct printer : boost::static_visitor<void>
{
printer(int indent) :indent(indent) {}
void operator()(const std::string& val) const
{
std::cout << std::string(indent, ' ') << val << std::endl;
}
void operator()(const std::map<std::string,Value>& val) const
{
for (const auto& pair : val)
{
std::cout << std::string(indent, ' ') << pair.first << ':' << std::endl;
boost::apply_visitor(printer(indent + 4), pair.second);
std::cout << std::string(indent, ' ') << std::endl;
}
}
int indent;
};
void print_map(const Value& val)
{
boost::apply_visitor(printer(0), val);
}
我想解析键值对,将字符串映射到字符串。因为我想在右手边支持 { ... }
块,所以我想出了一个简单的语法来开始
#include <boost/spirit/home/x3.hpp>
#include <boost/fusion/include/std_pair.hpp>
namespace grammar::map
{
using namespace boost::spirit::x3;
auto expression = rule<class expression, std::pair<std::string, std::string>>{"expression"};
auto lhs = *(~char_('='));
auto rest = *(~char_(';'));
auto block = '{' >> *expression >> '}';
auto expression_def = lhs >> '=' >> (block | rest) >> ';';
BOOST_SPIRIT_DEFINE(expression)
}
但是组合编译失败,除非我把expression的属性改成std::string
.
//! Transform a string into a key-value map of strings.
template <class M, class R>
requires InputRange<R>() && _ContainerLike<M>
&& Same<value_type_t<M>, std::pair<const std::string, std::string>>()
// && Assignable<std::pair<std::string, std::string>, value_type_t<M>>()
M parse(R&& range)
{
auto begin = rng::begin(range);
auto end = rng::end(range);
auto map = M{};
auto ret = x3::phrase_parse(begin, end, *grammar::map::expression, x3::space, map);
if (!ret)
throw std::runtime_error{"parse error"};
return map;
}
我收到错误
boost/spirit/home/x3/support/traits/move_to.hpp:62:18: error: cannot convert ‘std::remove_reference<std::pair<std::__cxx11::basic_string<char>, std::__cxx11::basic_string<char> >&>::type {aka std::pair<std::__cxx11::basic_string<char>, std::__cxx11::basic_string<char> >}’ to ‘char’ in assignment
dest = std::move(src);
位于
boost/spirit/home/x3/support/traits/move_to.hpp: In instantiation of ‘void boost::spirit::x3::traits::detail::move_to_plain(Source&&, Dest&, mpl_::false_) [with Source = std::pair<std::__cxx11::basic_string<char>, std::__cxx11::basic_string<char> >; Dest = char; mpl_::false_ = mpl_::bool_<false>]’:
如果尝试以下表达式,同样的事情会发生
auto pair = std::pair<std::string, std::string>{};
auto ret = x3::phrase_parse(begin, end, grammar::map::expression, x3::space, map);
几天前我就开始使用它,但不知道如何正确地进行操作。感谢您的帮助...:^)
使用 boost-1.60{-62} 和 gcc 6.1.1、6.2 以及较新的主干版本对其进行了测试。
你的问题是你定义了 expression
有一个 pair<string,string>
的属性,但是 lhs >> '=' >> (block|rest) >> ';'
的综合属性是 synt_attr=tuple<string,variant<vector<synt_attr>,string>>
基本上是 synt_attr=pair<string,variant<map_of_value_type_synt_attr,string>>
.因此,根据您想要的结果,您至少有两个选择:
将合成属性更改为
非常容易pair<string,string>
。使用x3::raw
指令 (running on WandBox):auto expression_def = lhs >> '=' >> (raw[block] | rest) >> ';';
更改您对
expression
的定义,使其具有与合成属性兼容的属性。这需要使用递归变体并使您需要访问已解析地图中的数据的方式复杂化,因为您需要创建访问者 (running on WandBox)。//A value is either a `string` or a `map<string,Value>` using Value = boost::make_recursive_variant<std::string,std::map<std::string,boost::recursive_variant_>>::type; ... struct printer : boost::static_visitor<void> { printer(int indent) :indent(indent) {} void operator()(const std::string& val) const { std::cout << std::string(indent, ' ') << val << std::endl; } void operator()(const std::map<std::string,Value>& val) const { for (const auto& pair : val) { std::cout << std::string(indent, ' ') << pair.first << ':' << std::endl; boost::apply_visitor(printer(indent + 4), pair.second); std::cout << std::string(indent, ' ') << std::endl; } } int indent; }; void print_map(const Value& val) { boost::apply_visitor(printer(0), val); }