【boost】使用serialization库序列化子类

boost.serialization库是一个很是强大又易用的序列化库,用于对象的保存与持久化等。网络

使用base_object能够在序列化子类的同时也序列化父类,以此得到足够的信息来从文件或网络数据中反序列化出子类。函数

最近在工做中却遇到这样一个问题,代码示例以下this

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(可能还会有其余子类),如今须要经过超类指针多态序列化子类spa

首先想到的办法是将超类serialize方法签名为virtual,经过多态调用子类serialize方法。可是这倒是不可能的,由于serialize方法是一个模板函数,不能为vitual方法,因此这条路不通。指针

无奈之下只好直接尝试一下:code

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后,终于找到了问题得解决办法,原文连接以下:xml

http://stackoverflow.com/questions/1332602/how-to-serialize-derived-template-classes-with-boost-serialize对象

http://stackoverflow.com/questions/8370727/error-serializing-an-abstract-class-with-boostblog

第一种是使用BOOST_SERIALIZATION_ASSUME_ABSTRACT配合archive的template register_type方法继承

第二种是直接使用BOOST_CLASS_EXPORT,导出全部继承链中的class

对于第二种方法直接使用

BOOST_CLASS_EXPORT(Field)
BOOST_CLASS_EXPORT(SubField<int>)

问题解决,输出为:

相关文章
相关标签/搜索