6

我已经阅读了大量类似的问题,但没有找到答案。我正在使用 Visual Studio 2010 和 boost 1.47。

这是代码,它是完整且可编译的:

#include "stdafx.h"

#include <string>
#include <sstream>

#include <boost/archive/text_oarchive.hpp>
#include <boost/archive/text_iarchive.hpp>

#include <boost/serialization/export.hpp>

using namespace std;

class BaseObject 
{
public:

    BaseObject(void) { };
    virtual ~BaseObject(void) { };

    template<class Archive>
      void serialize(Archive &ar, const unsigned int version)
      { /* nothing happens here */  };
};

class DerivedObject : public BaseObject
{
public:

    string text;

public:

    DerivedObject(void) { };
    ~DerivedObject(void) { };

    template<class Archive>
      void serialize(Archive &ar, const unsigned int version)
      {
          ar & text;
      };
};

BOOST_CLASS_EXPORT(DerivedObject)

int _tmain(int argc, _TCHAR* argv[])
{
    DerivedObject der;
    der.text = "Testing!";

    std::ostringstream os;
    boost::archive::text_oarchive oa(os);
    oa.register_type<DerivedObject>();

    // I made a DerivedObject, but I'm casting it to a BaseObject
    // as the serialization code should not have to know what type it is
    BaseObject *base = &der;
    // now serialize it
    oa << *base;

    printf("serialized: %s\r\n",os.str().c_str()); 

    return (0);
}

你可以看到它真的很简单,我已经添加了 BOOST_CLASS_EXPORT 和 oa.register_type 魔法,它应该确保 DerivdObject::serialize() 被调用,即使它不是虚拟方法.. 但仍然只有 serialize() 在BaseObject 被调用。可能是 Visual C++ 特有的问题?请指教?

4

3 回答 3

2

boost 序列化文档中所述,您需要告诉派生类调用基类序列化代码。只需像这样编写派生类序列化方法:

  template<class Archive>
  void serialize(Archive &ar, const unsigned int version)
  {
      ar & boost::serialization::base_object<BaseObject>(*this);
      ar & text;
  };
于 2013-03-28T11:53:01.930 回答
0

我没有在调试器或任何东西中尝试过这个,但它看起来可能是切片的情况。也许您可以通过修改代码以按指针或按引用而不是按值进行序列化来找出答案,就像这样...

BaseObject *base = &der;
oa << base;  // Serialize a pointer

...或者...

BaseObject& base = der;
oa << base;  // Serialize a reference
于 2012-03-16T06:15:09.803 回答
0

这不是严格的答案,只是一个笨拙的解决方法。

在基类中添加:

virtual void StreamToArchive(boost::archive::text_oarchive &oa) = 0;

然后定义一个宏 STREAMTOARCHIVE 并把它放到每一个派生类中。

#define STREAMTOARCHIVE void StreamToArchive(boost::archive::text_oarchive &oa) { oa << *this; }

然后在main中,替换

oa << base;

base.StreamToArchive(oa);

是的,我知道,它很丑,但是..它工作得很好,我只需要将 STREAMTOARCHIVE 宏放在派生类中......我可以忍受......

但是然后......将它解析回一个对象,现在这是另一回事......

编辑:将 'this' 更改为 '*this'

于 2012-03-16T21:11:06.543 回答