0 / 0 / 0
Регистрация: 15.12.2019
Сообщений: 2
1

Запись объекта класса с контейнером set в бинарный файл C++

15.12.2019, 23:52. Показов 1157. Ответов 3

Доброго времени суток! Пишу курсовую по теме "Работа с множествами". Хочу реализовать код через запись в бинарный файл, используя класс и контейнер set в нём. Но как раз и в контейнере заключена ошибка, которая все портит (-1073741819). Если пишу через, допустим, массив, то не вылетает. Поэтому вопрос: почему вылезает такая ошибка и можно ли её исправить? Если нельзя, то придется писать через вектора и/или обычные файлы.
Отмечу, что я еще не реализовал все функции, но и первые две не работают с set. Изначально удается создать пустое множество, но на следующем действии уже вылезает ошибка.

C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
#include "pch.h"
#include <iostream>
#include <string>
#include <fstream>
#include <set>
#include <iterator>
using namespace std;
 
class Set {
private:
    set<int> s;
    int elem;
public:
    unsigned int id;
    void setElement() {
        cout << "Enter element u want to add to the set: ";
        cin >> elem;
        s.insert(elem);
    }
 
    void getdata() {
        cout << "Set #" << id << " = { ";
        //copy(s.begin(), s.end(), ostream_iterator<int>(cout, " "));
        for (auto i : s) { cout << i << " "; }
        cout << "};" << endl;
    }
};
 
 
void create_set() {
    unsigned int *uid = new unsigned int;
    cout << "\nPlease enter the ID: ";
    cin >> *uid;
    Set S;
    ifstream I;
 
    I.open("Sets.txt", ios_base::app | ios_base::binary | ios_base::in);
    while (I.read((char*)&S, sizeof(Set))) {
        if (*uid == S.id) { cout << "ERROR. ID#" << *uid << " IS ALREADY IN USE. RETURNING TO MENU...\a\n\n"; delete uid; I.close(); return; }
    }
    ofstream O;
    O.open("Sets.txt", ios_base::app | ios_base::binary | ios_base::out);
    S.id = *uid;
    //S.getdata();
    O.write((char*)&S, sizeof(Set));
    O.close();
    delete uid;
    cout << "New set has been successfully created.\n\n";
}
void inclusion_element() {
    Set S;
    ifstream I;
    I.open("Sets.txt", ios_base::binary | ios_base::in);
    if (!I.is_open()) { cout << "ERROR. FILE WITH SETS DOES NOT EXIST.\a\n\n"; return; }
    unsigned int *uid = new unsigned int;
    cout << "\nPlease enter the ID: ";
    cin >> *uid;
    bool check = false;
    ofstream O;
    O.open("Sets.txt", ios_base::app | ios_base::binary | ios_base::out);
    while (I.read((char*)&S, sizeof(S))) {
        if (S.id == *uid) {
            S.setElement();
            O.write((char*)&S, sizeof(S));
            cout << "New element has been successfully entered to the set.\n\n";
            cout << "---------------------\n";
            check = true;
        }
    }
    delete uid;
    I.close();
    O.close();
    if (check == false)cout << "ERROR. SET DOES NOT EXIST.\a\n\n";
}
 
void display_sets() {
    Set S;
    ifstream F;
    F.open("Sets.txt", ios_base::binary);
    if (!F.is_open()) { cout << "ERROR. FILE WITH RECORDS DOES NOT EXIST.\a\n\n"; return; }
    cout << "\n\t| LIST OF ALL SETS |\n";
    while (F.read((char*)&S, sizeof(S))) {
        S.getdata();
        cout << "------------------------------------------" << endl;
    }
    F.close();
}
 
void delete_file() {
    cout << "Are you sure you want to clear the file? (Enter \"YES\" or \"NO\")\n";
    string choose;
    cin >> choose;
    if (choose == "YES" || choose == "yes" || choose == "Yes") { remove("Sets.txt"); cout << "File was successfully deleted.\n\n" << endl; return; }
    else if (choose == "NO" || choose == "no" || choose == "No") { return; }
    else { delete_file(); }
}
 
 
void menu() {
    char ch;
    do {
        cout << "\n\tMAIN MENU";
        cout << "\nD. DISPLAY SETS";
        cout << "\n1. CREATE AN EMPTY SET";
        cout << "\n2. INCLUSION OF AN ELEMENT IN A SET";
        cout << "\n3. EXCLUSION OF AN ELEMENT FROM THE SET";
        cout << "\n4. CHECKING THE ENTRY OF AN ELEMENT IN A SET";
        cout << "\n5. THE UNION OF TWO SETS";
        cout << "\n6. THE INTERSECTION OF TWO SETS";
        cout << "\n7. THE DIFFERENCE OF TWO SETS";
        cout << "\n8. CHECKING THE EQUALITY OF TWO SETS";
        cout << "\n9. CLEAR THE FILE";
        cout << "\n0. EXIT";
        cout << "\n\nEnter your choice (1-9): ";
        cin >> ch;
        switch (ch)
        {
        case 'D': display_sets(); break;
        case '1': create_set(); break;
        case '2': inclusion_element(); break;
            //case '3': exclusion_element(); break;
            //case '4': check_element(); break;
            //case '5': union_of_sets(); break;
            //case '6': intersection_of_sets(); break;
            //case '7': difference_of_sets(); break;
            //case '8': check_equality(); break;
        case '9': delete_file(); break;
        case '0': break;
        default: cout << "\a"; menu();
        }
    } while (ch != '0');
}
 
 
int main() {
    menu();
    return 0;
}
__________________
Помощь в написании контрольных, курсовых и дипломных работ, диссертаций здесь
0
Programming
Эксперт
94731 / 64177 / 26122
Регистрация: 12.04.2006
Сообщений: 116,782
15.12.2019, 23:52
Ответы с готовыми решениями:

Запись данных класса в бинарный файл
У меня был просто ввод-вывод в файл, но надо бинарный и мне выдает ошибку: &quot;error 'class...

Запись и чтения класса в бинарный файл
Здраствуйте. Можете сказать как в c# можно организовать запись и чтения класса с любыми данными (к...

Запись в файл объекта класса
class Object { private: int myN; std::vector&lt;std::string&gt; myLines; public: ...

Запись объекта класса в файл
Доброго времени суток! нужно сохранить объект вот такого класса: class Act { public: ...

3
16291 / 8847 / 2169
Регистрация: 30.01.2014
Сообщений: 15,291
16.12.2019, 00:37 2
Цитата Сообщение от Bleenok Посмотреть сообщение
почему вылезает такая ошибка и можно ли её исправить?
Допустим у вас есть вот такой класс:
C++
1
2
3
4
5
6
7
8
9
10
11
class A
{
public:
    A() : m_data(new int[10]())
    { }
    ~A()
    { delete[] m_data; }
 
private:
    int * m_data;
};
Если вы объект этого класса запишете в файл тем же способом, которым у вас пишется объект S, как вы думаете, что окажется в файле?

Цитата Сообщение от Bleenok Посмотреть сообщение
можно ли её исправить?
Можно. Только сначала вам надо разобраться в природе вещей. Подумайте над вопросом, который я задал.
0
0 / 0 / 0
Регистрация: 15.12.2019
Сообщений: 2
16.12.2019, 01:23  [ТС] 3
Цитата Сообщение от DrOffset Посмотреть сообщение
как вы думаете, что окажется в файле?
Хм, скорее всего там окажется набор случайных значений, привязанных к случайным местам памяти. То есть считать какую-либо информацию из файла будет невозможно, тк элементы массива будут разбросаны в памяти случайным образом.

Но это догадки.

Мне казалось, что если контейнер записан в класс, то и его элементы будут разбросаны в выделенной под класс памяти, а работать с ними будет возможно.
0
16291 / 8847 / 2169
Регистрация: 30.01.2014
Сообщений: 15,291
16.12.2019, 08:26 4
Цитата Сообщение от Bleenok Посмотреть сообщение
Хм, скорее всего там окажется набор случайных значений, привязанных к случайным местам памяти. То есть считать какую-либо информацию из файла будет невозможно, тк элементы массива будут разбросаны в памяти случайным образом.
Это очень размыто, но мыслите вы в целом верно. Я считаю, что пока вы досконально не поймете этот момент, нет смысла что-то учить дальше. На этом фундаменте потом будет строиться все понимание последующих концепций..

Цитата Сообщение от Bleenok Посмотреть сообщение
Мне казалось, что если контейнер записан в класс, то и его элементы будут разбросаны в выделенной под класс памяти, а работать с ними будет возможно.
Ну вот теперь вам самое время посмотреть на sizeof(std::set<int>).

И, самое главное, объяснить себе что такое тип в С++. Очень важно понимать что такое статическая типизация и почему статический тип не может менять свой размер во время выполнения программы.

Подсказка
Если бы std::set<int> содержал в себе всю память под элементы непосредственно, то добавлять в него элементы во время выполнения программы было бы нельзя (точнее можно, но количество добавляемых элементов было бы ограничено сверху, как это происходит в массиве). В любом случае у него был бы какой-то сопоставимый с количеством элементов sizeof. Он показывает размер статического типа. Статический тип известен только на этапе компиляции. Поэтому на самом деле std::set внутри очень похож на тот класс A, который я показал выше. Там точно также есть указатель в динамической памяти, через который управляется память под элементы, и когда вы записываете память переменной std::set в файл непосредственно, вы записываете значение этого указателя в файл и больше ничего. Поэтому при чтении, у вас получается ошибка исполнения. От того, что вы потом прочтете адрес из файла, по этому адресу память доступной не станет, и элементы в ней вдруг не появятся.
1
IT_Exp
Эксперт
87844 / 49110 / 22898
Регистрация: 17.06.2006
Сообщений: 92,604
16.12.2019, 08:26
Помогаю со студенческими работами здесь

Запись объекта класса в файл
Здравствуйте, может кто-нибудь помочь с данным вопросом? Как мне записать в файл строку, которая...

Запись в файл объекта класса
Добрый вечер,предположим есть подобный класс,с полями : const SIZE = 10; class myClass { ...

Запись и чтение объекта класса в файл
Доброго времени суток. Хочу разобраться с записью и чтением объекта класса в файл. Написал...

Запись и чтение объекта класса в файл и из файла
У меня есть std::list состоящий из объектов типа Person это база о людях. Нужно реализовать два...

Создание объекта класса и запись его в файл при нажатии кнопки
Добрый день. Стоит задача: при нажатии на кнопку должен создаваться объект моего класса, затем...

Запись в собственного класса бинарный файл собственного
есть Свой тип данных дробь. Надо реализовать запись и загрузку в\из бинарного файла. #ifndef...


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
4
Ответ Создать тему
Опции темы

КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2022, CyberForum.ru