Я хочу, чтобы моя программа сохраняла и считывала структуру конфигурации в файле JSON.
Однако у меня возникла проблема с созданием правильного файла JSON. Вероятно, проблема в наследовании.
Вывод JSON (неверный):
{
"config": {
"confVector": [
{
"common": "a"
},
{
"common": "b"
}
]
}
}
Ожидаемый (правильный) JSON:
{
"config": {
"confVector": [
{
"common": "a",
"a" : 1
},
{
"common": "b",
"b" : "b"
}
]
}
}
Код:
Базовая структура с общим элементом
struct Base
{
std::string common;
template <class Archive>
void serialize(Archive &ar)
{
ar(CEREAL_NVP(common));
}
};
Две специфические конструкции
struct A : public Base
{
int a;
template <class Archive>
void serialize(Archive &ar)
{
ar(cereal::make_nvp("Base", cereal::base_class<Base>(this)));
ar(cereal::make_nvp("a", a));
}
};
struct B : public Base
{
std::string b;
template <class Archive>
void serialize(Archive &ar)
{
ar(cereal::make_nvp("Base", cereal::base_class<Base>(this)));
ar(cereal::make_nvp("b", b));
}
};
struct Config
{
std::vector<Base> confVector;
template <class Archive>
void serialize(Archive &ar)
{
ar(CEREAL_NVP(confVector));
}
};
CEREAL_REGISTER_POLYMORPHIC_RELATION(Base, A)
CEREAL_REGISTER_POLYMORPHIC_RELATION(Base, B)
Основное: тестовое сохранение в файл json
int main()
{
std::string workPath = MAKE_STR(PLC_PROGRAM);
Config config;
A a;
a.a = 1;
a.common = "a";
B b;
b.b = "b";
b.common = "b";
config.confVector.push_back(a);
config.confVector.push_back(b);
std::ofstream outstream;
outstream.open(workPath + "/test.json");
{
cereal::JSONOutputArchive ar(outstream);
ar(cereal::make_nvp("config", config));
}
outstream.close();
}





Ваши объекты разрезаются.
std::vector<Base> confVector;
Это вектор объектов типа Base.
Если вы вставляете объекты, производные от Base, то копируется только базовая часть объекта.
A a;
B b;
config.confVector.push_back(a); // a is of Type A not Base
config.confVector.push_back(b); // b is of type B not Base
Итак, оба этих объекта нарезаются при помещении в вектор.
Вы можете хранить указатели на объект:
std::vector<Base> confVector;
...
config.confVector.push_back(&a);
config.confVector.push_back(&b);
Я решил эту проблему.
struct Base
{
Base() = default;
virtual ~Base() = default;
std::string common;
template <class Archive>
void serialize(Archive &ar)
{
ar(CEREAL_NVP(common));
}
};
struct A : public Base
{
A() = default;
A(int v)
{
a = v;
}
int a;
template <class Archive>
void serialize(Archive &ar)
{
ar(cereal::make_nvp("Base", cereal::base_class<Base>(this)));
ar(a);
}
};
struct B : public Base
{
B() = default;
B(std::string text)
{
b = text;
}
std::string b;
template <class Archive>
void serialize(Archive &ar)
{
ar(cereal::make_nvp("Base", cereal::base_class<Base>(this)));
ar(b);
}
};
struct Config
{
std::vector<std::shared_ptr<Base>> vector;
template <class Archive>
void serialize(Archive &ar)
{
ar(vector);
}
};
CEREAL_REGISTER_TYPE(A)
CEREAL_REGISTER_TYPE_WITH_NAME(B, "ClassB")
CEREAL_REGISTER_POLYMORPHIC_RELATION(Base, A)
CEREAL_REGISTER_POLYMORPHIC_RELATION(Base, B)
int main()
{
std::string workPath = "/home/user/"
{
std::ofstream os(workPath + "polymorphism_test.json");
cereal::JSONOutputArchive oarchive(os);
std::shared_ptr<Base> ptr1 = std::make_shared<A>(123);
std::shared_ptr<Base> ptr2 = std::make_shared<B>("foobar");
Config op;
op.vector.push_back(ptr1);
op.vector.push_back(ptr2);
oarchive(op);
}
{
std::ifstream is(workPath + "polymorphism_test.json");
cereal::JSONInputArchive iarchive(is);
Config ip;
iarchive(ip);
}
return 0;
}
Выход:
{
"value0": {
"value0": [
{
"polymorphic_id": 2147483649,
"polymorphic_name": "A",
"ptr_wrapper": {
"id": 2147483649,
"data": {
"Base": {
"common": ""
},
"value0": 123
}
}
},
{
"polymorphic_id": 2147483650,
"polymorphic_name": "ClassB",
"ptr_wrapper": {
"id": 2147483650,
"data": {
"Base": {
"common": ""
},
"value0": "foobar"
}
}
}
]
}
}
Я не могу найти хороший дубликат (уверен, что он где-то есть), но взгляните на нарезка объекта