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


    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/1332602/how-to-serialize-derived-template-classes-with-boost-serialize

    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>)

    问题解决,输出为:

      

  • 相关阅读:
    java基础之final
    java基础之finally(转)
    java 中 == 与 equals 的区别
    转载:日志分析
    eclipse配置Git
    Gitlab使用笔记:新建工程
    hadoop,spark的启动及DataNode无法启动的解决方法
    HTTP Status 500
    springmvc4.0配置ajax请求json格式数据
    jq load()方法中加载文件中元素事件绑定失效的问题
  • 原文地址:https://www.cnblogs.com/xiaosuiba/p/3444591.html
Copyright © 2020-2023  润新知