Сериализация сложных структур в C ++ - PullRequest
7 голосов
/ 19 августа 2010

Я пытаюсь сериализовать набор структур в C ++.Это прекрасно работает для всех данных, кроме вектора, содержащегося в моей структуре.Я могу записать данные на диск, а затем прочитать все данные обратно в память.Единственная проблема - когда я пытаюсь получить доступ к элементу вектора, я получаю ошибку сегментации.Мой код ниже.Любая помощь приветствуется.

Программа для записи на диск

int main {
  struct Student one;
  strcpy(one.FullName, "Ernestine Waller");
  strcpy(one.CompleteAddress, "824 Larson Drv, Silver Spring, MD 20910");
  one.Gender = 'F';
  one.LivesInASingleParentHome = true;
  one.grades.push_back(80);
  one.grades.push_back(90);

  ofstream ofs("fifthgrade.ros", ios::binary);
  ofs.write((char *)&one, sizeof(one));
  ofs.close();
}

Программа для чтения с диска

 int main {
    struct Student *two = (struct Student *) malloc(sizeof(struct Student));    
    ifstream ifs("fifthgrade.ros", ios::binary);
    //cout << "Size of struct: " << size << endl;
    ifs.read((char *)two, sizeof(struct Student));
    cout << "Student Name: " << two->FullName << endl;
    cout << "First Grade: " <<  two->grades[0] << endl;
    cout << "Second Grade: " << two->grades[1] << endl;
    ifs.close();
    return 0;
 }

Ответы [ 2 ]

7 голосов
/ 19 августа 2010

То, что вы делаете, это копирование непрерывной области памяти, в которой вы сохранили one и записали ее на диск. Это будет хорошо работать для простых типов данных (POD в C ++ jargan). Проблема с векторами заключается в том, что вектор представляет собой сложный объект с указателями на другие области памяти. Эти другие области памяти не существуют, когда вы десериализуете one в two, следовательно, ваша ошибка сегментации.

К сожалению, ярлыка нет, для выполнения работы потребуется написать некоторую форму для настраиваемого кода сериализации.

Как уже упоминалось, Повышенная сериализация может помочь. В качестве альтернативы выкатите свой собственный.

5 голосов
/ 19 августа 2010

Вероятно, самая простая версия сериализации вектора, которая имеет шанс на работу, выглядит примерно так:

your_stream << your_vector.size();
std::copy(your_vector.begin(), your_vector.end(),
     std::ostream_iterator<your_vector::value_type>(your_stream, "\n"));

Чтение обратно выглядит примерно так:

size_t size;
your_stream >> size;
vector_two.resize(size);
for (size_t i=0; i<size; i++)
    your_stream >> vector_two[i];

Обратите внимание, что это не особенно эффективно - в частности, оно (в основном) предполагает, что данные будут храниться в текстовом формате, который часто довольно медленный и занимает дополнительное место (но его легко читать, манипулировать и т. Д.). , сторонними программами, что часто бывает полезно).

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...