提升序列化输入流错误

use*_*806 5 c++ serialization boost

我正在开发一个简单的序列化类.我一直在输入流上抛出一个异常.我已经把我想要用简单的术语完成的下面的例子放在一起.

我有一个提升序列化的简单例子,我得到一个例外:

#include <boost/serialization/serialization.hpp> 
#include <boost/archive/text_oarchive.hpp> 
#include <boost/archive/text_iarchive.hpp> 
#include <boost/serialization/export.hpp> 
#define NVP(X) X 

class base { 
public: 
friend class boost::serialization::access; 
base (){ v1 = 10;} 
int v1; 
template<class Archive> 
void serialize(Archive & ar, const unsigned int file_version) 
{ 

    ar & NVP(v1); 
} 
virtual void bla()=0; 
}; 


class derived : public base { 
public: 
friend class boost::serialization::access; 
int v2 ; 
derived() { v2 = 100;} 
template<class Archive> 
void serialize(Archive & ar, const unsigned int file_version){ 
    boost::serialization::base_object<base>(* this); 
    ar & NVP(v2); 
} 

virtual void bla(){};
};
BOOST_CLASS_EXPORT(base);
BOOST_CLASS_EXPORT_GUID(derived, "derived");

int main ( )
{
std::stringstream ss;
boost::archive::text_oarchive ar(ss);
base *b = new derived();
ar << NVP(b);
std::cout << ss.str()<<std::endl;



std::istringstream ssi;
base *b1 = new derived();
{
boost::archive::text_iarchive ar1(ssi);
ar1 >> b1;
}
//std::cout << ssi.str();
std::cout << "v1: " << b1->v1 << std::endl;

}
Run Code Online (Sandbox Code Playgroud)

我得到的例外是:

terminate called after throwing an instance of 'boost::archive::archive_exception'
  what():  input stream error
Run Code Online (Sandbox Code Playgroud)

任何帮助,将不胜感激.

seh*_*ehe 4

您正在从空流中读取:

std::istringstream ssi;
// ...
boost::archive::text_iarchive ar1(ssi);
Run Code Online (Sandbox Code Playgroud)

另外,你泄漏了这个对象:

base *b1 = new derived();
Run Code Online (Sandbox Code Playgroud)

这是一个固定的例子,注意:

  • 在使用流数据之前关闭档案是非常好的做法/很重要
  • BOOST_CLASS_EXPORT_GUID(derived, "derived")不添加任何东西BOOST_CLASS_EXPORT(derived)
  • 您可以有条件地打印 v2:

    if (auto* d = dynamic_cast<derived*>(b1))
        std::cout << "v2: " << d->v2 << std::endl;
    
    Run Code Online (Sandbox Code Playgroud)
  • 我用它bla()作为示例来打印值

  • NVP()那里有点不确定。为什么不将其保留给非标记档案(即 XML 之外的档案)呢?如果您打算支持 XML,只需使用等BOOST_SERIALIZATION_NVPboost::serialization::make_nvp

  • std::cout << "v2: " << b1->v2 << std::endl;完全不合适

  • 只需初始化b1为 null,这样就不会泄漏;记住释放所有指针(使用智能指针!)

  • public:你的类型中和的混合friend并没有多大意义

Live On Coliru

#include <boost/serialization/serialization.hpp>
#include <boost/archive/text_oarchive.hpp>
#include <boost/archive/text_iarchive.hpp>
#include <boost/serialization/export.hpp>
#include <sstream>

class base {
  public:
    base(int v1) : v1(v1) {}
    virtual void bla() const = 0;

  private:
    friend class boost::serialization::access;

    template <class Archive> void serialize(Archive &ar, unsigned /*int const file_version*/) {
        ar & BOOST_SERIALIZATION_NVP(v1);
    }
  protected:
    int v1;
};

class derived : public base {
  public:
    derived(int v1 = 10, int v2 = 100) : base(v1), v2(v2) {}
    virtual void bla() const {
        std::cout << "v1: " << v1 << ", v2: " << v2 << "\n";
    }

  private:
    friend class boost::serialization::access;
    int v2;
    template <class Archive> void serialize(Archive &ar, unsigned /*int const file_version*/) {
        boost::serialization::base_object<base>(*this);
        ar & BOOST_SERIALIZATION_NVP(v2);
    }
};

BOOST_CLASS_EXPORT(base)
BOOST_CLASS_EXPORT(derived)

int main() {
    std::stringstream ss;
    {
        boost::archive::text_oarchive ar(ss);
        base *b = new derived();
        ar << boost::serialization::make_nvp("base", b);

        delete b; // TODO use RAII instead
    }

    std::cout << ss.str() << std::endl;
    base *deserialized = nullptr;
    {
        boost::archive::text_iarchive ar1(ss);
        ar1 >> boost::serialization::make_nvp("base", deserialized);
    }

    deserialized->bla();
    delete deserialized;
}
Run Code Online (Sandbox Code Playgroud)

印刷

22 serialization::archive 12 0 7 derived 1 0
0 100

v1: 10, v2: 100
Run Code Online (Sandbox Code Playgroud)