我正在使用 boost spirit 将一些文本文件解析为一个数据结构,现在我开始从这个数据结构中生成文本(使用 spirit karma)。
对数据结构的一种尝试是 boost::fusion::map (如对 这个问题的回答中所建议的那样)。但是,虽然我可以使用 boost::spirit::qi::parse() 并轻松获取其中的数据,但当我尝试使用 karma 从中生成文本时,我失败了。
下面是我的尝试(尤其是“map_data”类型)。在阅读和尝试了其他融合类型之后,我发现了 boost::fusion::vector 和 BOOST_FUSION_DEFINE_ASSOC_STRUCT。我成功地用它们生成了输出,但它们似乎并不理想:在向量中,你不能使用名称访问成员(它就像一个元组)——而在另一个解决方案中,我认为我不需要两种方式(成员名称和密钥类型)来访问成员。
#include <iostream>
#include <string>
#include <boost/spirit/include/karma.hpp>
#include <boost/fusion/include/map.hpp>
#include <boost/fusion/include/make_map.hpp>
#include <boost/fusion/include/vector.hpp>
#include <boost/fusion/include/as_vector.hpp>
#include <boost/fusion/include/transform.hpp>
struct sb_key;
struct id_key;
using boost::fusion::pair;
typedef boost::fusion::map
< pair<sb_key, int>
, pair<id_key, unsigned long>
> map_data;
typedef boost::fusion::vector < int, unsigned long > vector_data;
#include <boost/fusion/include/define_assoc_struct.hpp>
BOOST_FUSION_DEFINE_ASSOC_STRUCT(
(), assocstruct_data,
(int, a, sb_key)
(unsigned long, b, id_key))
namespace karma = boost::spirit::karma;
template <typename X>
std::string to_string ( const X& data )
{
std::string generated;
std::back_insert_iterator<std::string> sink(generated);
karma::generate_delimited ( sink, karma::int_ << karma::ulong_, karma::space, data );
return generated;
}
int main()
{
map_data d1(boost::fusion::make_map<sb_key, id_key>(234, 35314988526ul));
vector_data d2(boost::fusion::make_vector(234, 35314988526ul));
assocstruct_data d3(234,35314988526ul);
std::cout << "map_data as_vector: " << boost::fusion::as_vector(d1) << std::endl;
//std::cout << "map_data to_string: " << to_string(d1) << std::endl; //*FAIL No 1*
std::cout << "at_key (sb_key): " << boost::fusion::at_key<sb_key>(d1) << boost::fusion::at_c<0>(d1) << std::endl << std::endl;
std::cout << "vector_data: " << d2 << std::endl;
std::cout << "vector_data to_string: " << to_string(d2) << std::endl << std::endl;
std::cout << "assoc_struct as_vector: " << boost::fusion::as_vector(d3) << std::endl;
std::cout << "assoc_struct to_string: " << to_string(d3) << std::endl;
std::cout << "at_key (sb_key): " << boost::fusion::at_key<sb_key>(d3) << d3.a << boost::fusion::at_c<0>(d3) << std::endl;
return 0;
}
包括注释行会给出很多页面的编译错误,其中值得注意的是:从 'boost::fusion::pair' 到 'double' 的参数 1 没有已知的转换 'boost::fusion 的参数 1 没有已知的转换::pair'到'浮动'</p>
可能是 to_string 需要 map_data 的值,而不是对?虽然我不擅长模板,但我尝试通过以下方式使用变换从地图中获取矢量
template <typename P>
struct take_second
{
typename P::second_type operator() (P p)
{
return p.second;
}
};
// ... inside main()
pair <char, int> ff(32);
std::cout << "take_second (expect 32): "
<< take_second<pair<char,int>>()(ff) << std::endl;
std::cout << "transform map_data and to_string: "
<< to_string(boost::fusion::transform(d1, take_second<>())); //*FAIL No 2*
但是我不知道在实例化 take_second 时我应该给出什么类型,无论如何我认为必须有一种更简单的方法来获取(迭代)地图的值(有吗?)
如果您回答了这个问题,还请就使用 ASSOC_STRUCT 还是使用地图更好地发表您的意见。