У меня есть четыре класса (A
,B
,C
и D
), следующих классическому ромбовидному узору, и класс Container
, содержащий unique_ptr<A>
. Я хочу сериализовать эти классы с помощью библиотеки сериализации cereal.
struct A {int f1; int f2; int f3}
struct B : public virtual A {
template<typename Archive>
inline void save(Archive& ar) const {
std::cerr << "Saving Obj: " << this << std::endl;
std::cerr << "This: " << &(this->f1) << " "
<< &(this->f2) << " " << &(this->f3) << std::endl;
std::cerr << "This: " << this->f1 << " "
<< this->f2 << " " << this->f3 << std::endl;
};
}
};
struct C : public virtual A {};
struct D : public B, public C {};
#include <cereal/archives/binary.hpp>
CEREAL_REGISTER_TYPE(B);
CEREAL_REGISTER_TYPE(C);
CEREAL_REGISTER_TYPE(D);
struct Container {
std::unique_ptr<A> obj;
template<typename Archive>
inline void save(Archive& ar) const {
std::cerr << "Saving Container" << std::endl;
std::cerr << "Obj Addr: " << obj.get() << std::endl;
std::cerr << "Obj: " << &(obj->f1) << " " << &(obj->f2)
<< " " << &(pq->f3) << std::endl;
std::cerr << "Obj: " << " " << pq->sq_count << " " << pq->sq_bits
<< " " << pq->dim << std::endl;
ar(obj); // Call serialization for obj, ie B.save(...)
}
}
Все классы имеют функции зерновых save
и load
, но я включил их только для B
и Container
, так как в этом примере используются только они.
Я использую эти классы следующим образом:
std::unique_ptr<A> obj(new B);
obj->f1 = 8;
obj->f2 = 8;
obj->f3 = 128;
std::unique_ptr<Container> db(new Container);
db.obj = std::move(obj);
std::ofstream out_file(out_filename);
cereal::BinaryOutputArchive out_archive(out_file);
out_archive(db);
И я получаю следующий вывод:
Saving Container
Obj Addr: 0x23d2128
Obj: 0x23d2130 0x23d2134 0x23d2138 // Fields adresses (f1,f2,f3)
Obj: 8 8 128 // Fields values
Saving Obj: 0x23d2128 // Same object
This: 0x23d2118 0x23d211c 0x23d2120 // Different field adresses !
This: 4293296 0 37569440 // Garbage
Мой вопрос: вероятно ли, что это ошибка в хлопьях или есть что-то, чего я не получаю с виртуальным наследованием?
Ожидается ли, что адреса полей данного объекта когда-либо изменятся в программе на C++?
ar(obj)
звонитьB::save(ar)
? - person aschepler   schedule 13.02.2016obj
— этоstd::unique_ptr<A>
, а наиболее производным типом*obj
являетсяA
. Где вмешиваетсяB
? - person aschepler   schedule 13.02.2016*obj
должен быть B, я неправильно понял вопрос. - person Xion345   schedule 13.02.2016