boost.serialization库是一个非常强大又易用的序列化库,用于对象的保存与持久化等。
使用base_object可以在序列化子类的同时也序列化父类,以此获得足够的信息来从文件或网络数据中反序列化出子类。
最近在工作中却遇到这样一个问题,代码示例如下
struct Field { friend class boost::serialization::access; template<class Archive> void serialize(Archive & ar, const unsigned int version) { ar & boost::serialization::make_nvp("FieldImpl", serializeString); } string serializeString; }; template<typename ValueType> struct SubField : Field { typedef ValueType type; friend class boost::serialization::access; template<class Archive> void serialize(Archive & ar, const unsigned int version) { ar & boost::serialization::make_nvp("CommonField", boost::serialization::base_object<Field>(*this)); ar & boost::serialization::make_nvp("data", data); } type data; };
有超类Field与模板子类SubField(可能还会有其他子类),现在需要通过超类指针多态序列化子类
首先想到的办法是将超类serialize方法签名为virtual,通过多态调用子类serialize方法。但是这却是不可能的,因为serialize方法是一个模板函数,不能为vitual方法,所以这条路不通。
无奈之下只好直接尝试一下:
shared_ptr<SubField<int> > spSub(new SubField<int>()); spSub->data = 10; spSub->serializeString = "sub"; shared_ptr<Field> spField(new Field()); spField = boost::static_pointer_cast<Field>(spSub); std::stringstream ss; boost::archive::xml_oarchive ar(ss, boost::archive::archive_flags::no_header); ar << boost::serialization::make_nvp("Field", spField); std::cout << ss.str();
不出所料,运行时抛出了异常。
重新阅读手册及serialization文档,意外发现几个关键词,BOOST_SERIALIZATION_ASSUME_ABSTRACT和BOOST_CLASS_EXPORT,继续求助sof后,终于找到了问题得解决办法,原文链接如下:
http://stackoverflow.com/questions/8370727/error-serializing-an-abstract-class-with-boost
第一种是使用BOOST_SERIALIZATION_ASSUME_ABSTRACT配合archive的template register_type方法
第二种是直接使用BOOST_CLASS_EXPORT,导出所有继承链中的class
对于第二种方法直接使用
BOOST_CLASS_EXPORT(Field) BOOST_CLASS_EXPORT(SubField<int>)
问题解决,输出为: