Форум программистов, компьютерный форум CyberForum.ru
Наши страницы

С++ для начинающих

Войти
Регистрация
Восстановить пароль
 
 
murtukov
8 / 8 / 3
Регистрация: 30.01.2013
Сообщений: 99
#1

Не могу найти ошибку. Запись/чтение из файла - C++

06.10.2013, 00:45. Просмотров 934. Ответов 22
Метки нет (Все метки)

Простите за довольно таки большой код, но обратиться больше некуда. Урезать не получилось, постараюсь на словах дать общую концепцию:

Есть абстрактный базовый класс Person и два его наследника: Student и Laborer.
Также есть класс Database, который хранит в себе эти объекты с помощью контейнера std::list<Person const *>.

Вот пример кода, работающий без ошибок:

Student s1; - создать объект
s1.askData(); - запросить данные у пользователя

Laborer l1; - создать объект
l1.askData(); - запросить данные у пользователя

Database db; - создать базу объектов
db.add(&s1); - добавить s1 в базу
db.add(&l1); - добавить l1 в базу

db.printList(); - вывести все данные всех объектов из базы

Так же в базу можно добавлять объекты сразу, без предварительного создания объектов:

Объект определяется количеством аргументов
db.add("Martin", 311017); - добавляет в базу объект Student с именем Martin и ID 311017
db.add("Klark", "4943334325", "14B"); - добавляет в базу объект Laborer с именем Klark, номером телефона и бюро.

Вы наверное спросите себя, зачем я это все здесь пишу, если все работает? Проблема в том, что надо реализовать сохранение всей базы db в файл и считывание из этого файла. Код мною уже написан, но не работает.

Метод сохраняющий базу называется void Database::saveToFile();
Для считывания есть метод void Database::readFromFile();

Собственно ошибка кроется где-то в них. Файл создается и информация вроде как сохраняется, но вот считать не получается. Прошу помочь разобраться.

Полный код программы:

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
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
#include <iostream>
#include <list>
#include <typeinfo>
#include <stdlib.h>
#include <fstream>
using namespace std;
 
class Person;
class Student;
class Laborer;
typedef list<Person const *> Perslist;
 
enum PersType { tstudent, tLaborer };
////////////////////////////////////////////////////////////////
class Person
{
protected:
    string name;
 
public:
    Person()
    {}
 
    Person(const char *n)
    { name = n; }
 
    virtual ~Person()
    {}
 
    string retName() const
    { return name; }
 
    void getName(const char *n)
    { name = n; }
 
    void askData()
    { cout << "  Enter a name: "; cin >> name; }
 
    void printStream(ostream& os) const
    { os << name << endl; }
 
    virtual void print() const
    { cout << "  Name: " << name << endl; }
 
    bool operator == (string s)
    {
        if(name == s)
            return 1;
        else
            return 0;
    }
 
    virtual PersType getType() const;
 
    virtual void pure() = 0; // pure virtual func
};
////////////////////////////////////////////////////////////////
class Student : public Person
{
    unsigned int matricul;
 
public:
    Student() {}
 
    Student(string const n, unsigned int m)
    {
        Person::name = n;
        matricul = m;
    }
 
    void askData()
    {
        Person::askData();
        cout << "  Enter a matriculation: ";
        cin >> matricul;
    }
 
    void printStream(ostream& os) const
    {
        Person::printStream(os);
        cout << matricul << endl << endl;
    }
 
    void print() const
    {
        Person::print();
        cout << "  Matriculation: " << matricul << endl << endl;
    }
    void pure() {}
};
////////////////////////////////////////////////////////////////
class Laborer : public Person
{
    string tel;
    string buro;
public:
    Laborer() {}
 
    Laborer(string const n, string const t,
                string const b)
    {
        Person::name = n;
        tel = t;
        buro = b;
    }
 
    void askData()
    {
        Person::askData();
        cout << "  Enter a tel: ";
        cin >> tel;
        cout << "  Enter a buro: ";
        cin >> buro;
    }
 
    void printStream(ostream& os)
    {
        Person::printStream(os);
        os << "  Telefon: " << tel << endl
           << "  Buro: " << buro << endl << endl;
    }
 
    void print() const
    {
        Person::print();
        cout << "  Telefon: " << tel << endl
             << "  Buro: " << buro << endl << endl;
    }
 
    void pure() {}
};
////////////////////////////////////////////////////////////////
class Database
{
    Perslist plist;
 
public:
    ~Database()
    {
        plist.clear();
    }
 
    void add(Person const *p)
    {
        plist.push_back(p);
    }
 
    void add(string const name, unsigned const matr)
    {
        Student *s = new Student(name, matr);
        plist.push_back(s);
    }
 
    void add(string const name, string const tel,
             string const buro)
    {
        Laborer *m = new Laborer(name, tel, buro);
        plist.push_back(m);
    }
 
    void printList()
    {
        Perslist::iterator it = plist.begin();
        for (int i=0; i < plist.size(); ++i, ++it)
        {
            (*it)->print();
        }
    }
 
    const Person* find(string const n)
    {
 
        Perslist::iterator it = plist.begin();
        for (int i=0; i < plist.size(); ++i, ++it)
        {
            if( (*it)->retName() == n )
            {
                return *it;
            }
        }
    }
 
    int howMuchS()
    {
        int n = 0;
        string stud = "7Student";
        Perslist::iterator it = plist.begin();
        for(int i=0; i < plist.size(); ++i, ++it)
        {
            if(typeid(**it).name() == stud)
                ++n;
        }
        return n;
    }
 
    void saveToFile(const char *filename)
    {
        PersType ptype;
        int size;
        fstream ofile;
        ofile.open(filename, ios::app | ios::out |
                                        ios::binary);
 
        Perslist::iterator it = plist.begin();
 
        for(int i=0; i < plist.size(); ++i, ++it)
        {
            ptype = (*it)->getType();
            ofile.write((char*)&ptype, sizeof(ptype));
 
            switch(ptype)           // найти его размер
            {
                case tstudent:   size  = sizeof(Student); break;
                case tLaborer: size  = sizeof(Laborer); break;
            }
            ofile.write((char*)*it, size);
        }
        ofile.close();
 
    }
 
    void readFromFile(const char *filename)
    {
        PersType ptype;
        int size;
        fstream ifile;
        Person *p;
 
        ifile.open(filename, ios::binary);
 
        Perslist::iterator it = plist.begin();
        Perslist::iterator endit = plist.end();
        while(it != endit)
        {
            ifile.read((char*)&ptype, sizeof(ptype));
 
            switch(ptype)
            {
            case tstudent:
                p = new Student;
                size = sizeof(Student);
                ifile.read((char*)p, size);
                plist.push_back(p);
                break;
            case tLaborer:
                p = new Laborer;
                size = sizeof(Laborer);
                ifile.read((char*)p, size);
                plist.push_back(p);
                break;
            default:
                cout << "Unknown format" << endl;
            }
        }
        ifile.close();
    }
};
////////////////////////////////////////////////////////////////
 PersType Person::getType() const
 {
        if(typeid(*this) == typeid(Student))
            return tstudent;
        else if(typeid(*this) == typeid(Laborer))
            return tLaborer;
        else
        {
            cerr << "\nWrong type of Person";
            exit(1);
        }
}
//--------------------------------------------------------------
bool operator == (string& s, Person& p)
{
    if(s == p.retName())
        return 1;
    else
        return 0;
}
//--------------------------------------------------------------
void operator << (ostream& os, Person& p)
{
    os << p.retName();
}
////////////////////////////////////////////////////////////////
int main()
{
    Database db;
 
    db.add("Kristina", 241089);
    db.add("Martina", 311017);
    db.add("Kirill", 1111);
    db.add("Harold", "0170-4081883", "Vertrieb");
 
    db.saveToFile("Proverka.dat");
 
    db.printList();
    return 0;
}
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
Similar
Эксперт
41792 / 34177 / 6122
Регистрация: 12.04.2006
Сообщений: 57,940
06.10.2013, 00:45
Здравствуйте! Я подобрал для вас темы с ответами на вопрос Не могу найти ошибку. Запись/чтение из файла (C++):

НЕ могу найти ошибку.Чтение из файла - C++
// Лаба №5.cpp: определяет точку входа для консольного приложения. // #include &quot;stdafx.h&quot; #include &quot;math.h&quot; #include &quot;stdio.h&quot; ...

Чтение и запись файла - C++
Странная ситуация читаю файл в вектор. Провожу над ним эксперимент, возвращаю точную копию считанного файла в новый файл с другим...

Чтение/запись файла - C++
Вот сделал такую прогу. Очень простая. Создает файл, где есть 100 полей. Каждое поле это строка из индекса, названия, кол-ва (например...

запись\чтение из файла - C++
У меня несколько вопросов по поводу запись\чтение из файла Как это вообще работает?) Чтобы потом читать из файла надо забивать каждое...

Чтение и запись из файла - C++
Пользуюсь code::blocks. Перерыл весь гугл, так и не понял, как что-то читать или записыват в файл при помощи библиотеки &lt;fstream&gt;. Можете...

Запись и чтение из файла - C++
Доброго время суток! Так от балди решыл набросать прогу которая создаёт файл записывает туда 11 рандомных чисел,потом з помощю ifstream я...

22
alsav22
5421 / 4816 / 442
Регистрация: 04.06.2011
Сообщений: 13,587
06.10.2013, 00:50 #2
Если в объектах, которые пишите в файл в бинарном режиме, есть объекты (типа string), то ничего не получится. Объект string и строка, с которой он связан - не одно и тоже. Пишите поля в текстовом режие.
1
murtukov
8 / 8 / 3
Регистрация: 30.01.2013
Сообщений: 99
06.10.2013, 00:57  [ТС] #3
alsav22,
Как тогда сохранить объект с полем типа string в файл?
Что вы подразумеваете под "Пишите поля в текстовом режиме"?
0
alsav22
5421 / 4816 / 442
Регистрация: 04.06.2011
Сообщений: 13,587
06.10.2013, 00:59 #4
Цитата Сообщение от murtukov Посмотреть сообщение
Что вы подразумеваете под "Пишите поля в текстовом режиме"?
Не бинарные методы чтения\записи: <<, >>, getline() и пр. Какой выбрать - сами смотрите, в код не вникал.
1
murtukov
8 / 8 / 3
Регистрация: 30.01.2013
Сообщений: 99
06.10.2013, 01:22  [ТС] #5
alsav22, я заменил все строки типа string на char, ничего не изменилось, копирование не работает. Дело в ошибочном коде
0
alsav22
5421 / 4816 / 442
Регистрация: 04.06.2011
Сообщений: 13,587
06.10.2013, 01:32 #6
Цитата Сообщение от murtukov Посмотреть сообщение
Дело в ошибочном коде
Понятно, что в коде (в чём же ещё?). Ошибка может быть и не одна. Я же написал, что в код не вникал. Покажите как сделали.
0
murtukov
8 / 8 / 3
Регистрация: 30.01.2013
Сообщений: 99
06.10.2013, 01:40  [ТС] #7
alsav22,

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
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
#include <iostream>
#include <list>
#include <typeinfo>
#include <stdlib.h>
#include <fstream>
#include <string.h>
using namespace std;
 
class Person;
class Student;
class Mitarbeiter;
typedef list<Person const *> Perslist;
 
enum PersType { tstudent, tmitarbeiter };
////////////////////////////////////////////////////////////////
class Person
{
protected:
    char name[100];
 
public:
    Person()
    {}
 
    Person(const char *n)
    {
        strcpy(name, n);
    }
 
    virtual ~Person()
    {}
 
    const char* retName() const
    {
        return name;
    }
 
    void getName(const char *n)
    {
        strcpy(name, n);
    }
 
    void askData()
    {
        cout << "  Enter a name: ";
        cin >> name;
    }
 
    void printStream(ostream& os) const
    {
        os << name << endl;
    }
 
    virtual void print() const
    {
        cout << "  Name: " << name << endl;
    }
 
    bool operator == (string s)
    {
        if(name == s)
            return 1;
        else
            return 0;
    }
 
    virtual PersType getType() const;
 
    virtual void pure() = 0; // pure virtual func
};
////////////////////////////////////////////////////////////////
class Student : public Person
{
    unsigned int matrikel;
 
public:
    Student() {}
 
    Student(char const *n, unsigned int m)
    {
        strcpy(Person::name, n);
        matrikel = m;
    }
 
    void askData()
    {
        Person::askData();
        cout << "  Enter a matriculation: ";
        cin >> matrikel;
    }
 
    void printStream(ostream& os) const
    {
        Person::printStream(os);
        cout << matrikel << endl << endl;
    }
 
    void print() const
    {
        Person::print();
        cout << "  Matriculation: " << matrikel << endl << endl;
    }
    void pure() {}
};
////////////////////////////////////////////////////////////////
class Mitarbeiter : public Person
{
    char tel[100];
    char buro[100];
public:
    Mitarbeiter() {}
 
    Mitarbeiter(char const *n, char const *t,
                char const *b)
    {
        strcpy(Person::name, n);
        strcpy(tel, t);
        strcpy(buro, b);
    }
 
    void askData()
    {
        Person::askData();
        cout << "  Enter a tel: ";
        cin >> tel;
        cout << "  Enter a buro: ";
        cin >> buro;
    }
 
    void printStream(ostream& os)
    {
        Person::printStream(os);
        os << "  Telefon: " << tel << endl
           << "  Buro: " << buro << endl << endl;
    }
    void print() const
    {
        Person::print();
        cout << "  Telefon: " << tel << endl
             << "  Buro: " << buro << endl << endl;
    }
 
    void pure() {}
};
////////////////////////////////////////////////////////////////
class Database
{
    Perslist plist;
 
public:
    ~Database()
    {
        plist.clear();
    }
 
    void add(Person const *p)
    {
        plist.push_back(p);
    }
 
    void add(char const *name, unsigned const matr)
    {
        Student *s = new Student(name, matr);
        plist.push_back(s);
    }
 
    void add(char const *name, char const *tel,
             char const *buro)
    {
        Mitarbeiter *m = new Mitarbeiter(name, tel, buro);
        plist.push_back(m);
    }
 
    void printList()
    {
        Perslist::iterator it = plist.begin();
        for (int i=0; i < plist.size(); ++i, ++it)
        {
            (*it)->print();
        }
    }
 
    const Person* find(string const n)
    {
 
        Perslist::iterator it = plist.begin();
        for (int i=0; i < plist.size(); ++i, ++it)
        {
            if( (*it)->retName() == n )
            {
                return *it;
            }
        }
    }
 
    int howMuchS()
    {
        int n = 0;
        string stud = "7Student";
        Perslist::iterator it = plist.begin();
        for(int i=0; i < plist.size(); ++i, ++it)
        {
            if(typeid(**it).name() == stud)
                ++n;
        }
        return n;
    }
 
    void saveToFile(const char *filename)
    {
        PersType ptype;
        int size;
        fstream ofile;
        ofile.open(filename, ios::app | ios::out |
                                        ios::binary);
 
        Perslist::iterator it = plist.begin();
 
        for(int i=0; i < plist.size(); ++i, ++it)
        {
            ptype = (*it)->getType();
            ofile.write((char*)&ptype, sizeof(ptype));
 
            switch(ptype)           // найти его размер
            {
                case tstudent:   size  = sizeof(Student); break;
                case tmitarbeiter: size  = sizeof(Mitarbeiter); break;
            }
            ofile.write((char*)*it, size);
        }
        ofile.close();
 
    }
 
    void readFromFile(const char *filename)
    {
        PersType ptype;
        int size;
        fstream ifile;
        Person *p;
 
        ifile.open(filename, ios::binary);
 
        Perslist::iterator it = plist.begin();
        Perslist::iterator endit = plist.end();
        while(it != endit)
        {
            ifile.read((char*)&ptype, sizeof(ptype));
 
            switch(ptype)
            {
            case tstudent:
                p = new Student;
                size = sizeof(Student);
                ifile.read((char*)p, size);
                plist.push_back(p);
                break;
            case tmitarbeiter:
                p = new Mitarbeiter;
                size = sizeof(Mitarbeiter);
                ifile.read((char*)p, size);
                plist.push_back(p);
                break;
            default:
                cout << "Unknown format" << endl;
            }
        }
        ifile.close();
    }
};
////////////////////////////////////////////////////////////////
 PersType Person::getType() const
 {
        if(typeid(*this) == typeid(Student))
            return tstudent;
        else if(typeid(*this) == typeid(Mitarbeiter))
            return tmitarbeiter;
        else
        {
            cerr << "\nWrong type of Person";
            exit(1);
        }
}
//--------------------------------------------------------------
bool operator == (string& s, Person& p)
{
    if(s == p.retName())
        return 1;
    else
        return 0;
}
//--------------------------------------------------------------
void operator << (ostream& os, Person& p)
{
    os << p.retName();
}
////////////////////////////////////////////////////////////////
int main()
{
 
    Database db;
 
    db.add("Kristina", 241089);
    db.add("Martina", 311017);
    db.add("Kirill", 1111);
    db.add("Harold", "0170-4081883", "Vertrieb");
 
    db.saveToFile("Proverka.dat");
 
//    db.readFromFile("Proverka.dat");
    db.printList();
    return 0;
}
0
novi4ok
551 / 504 / 8
Регистрация: 23.07.2009
Сообщений: 2,359
Записей в блоге: 1
06.10.2013, 01:51 #8
а ты провто для интереса возьми и выведи куда-нибудь себе значение sizeof(Student). будешь очень удивлен.

Добавлено через 58 секунд
нужно нормально сериализировать и десериализировать объекты. они "от рождения" этого делать не умеют
0
alsav22
5421 / 4816 / 442
Регистрация: 04.06.2011
Сообщений: 13,587
06.10.2013, 02:01 #9
murtukov, моё предлоожение было: не использоавть бинарное чтение\запись.

Добавлено через 1 минуту
А если использовать, то тут нужно подумать.
0
murtukov
8 / 8 / 3
Регистрация: 30.01.2013
Сообщений: 99
06.10.2013, 02:01  [ТС] #10
alsav22, я понял, но это не решение, я уже сохранял объекты со строковыми полями (char*) в файл и все прекрасно работало. Могу показать рабочий пример
0
novi4ok
551 / 504 / 8
Регистрация: 23.07.2009
Сообщений: 2,359
Записей в блоге: 1
06.10.2013, 02:03 #11
Цитата Сообщение от alsav22 Посмотреть сообщение
murtukov, моё предлоожение было: не использоавть бинарное чтение\запись.
это куда? просто понимать нужно, что именно ты пишешь.
0
murtukov
8 / 8 / 3
Регистрация: 30.01.2013
Сообщений: 99
06.10.2013, 02:03  [ТС] #12
novi4ok, novi4ok, видимо мне надо почитать о сериализации и десериализации объектов. Никогда об этом не слышал прежде. Есть хорошие источники?
0
alsav22
5421 / 4816 / 442
Регистрация: 04.06.2011
Сообщений: 13,587
06.10.2013, 02:03 #13
Я бы вообще так не делал. Где-то работает, а где-то не будет. Но это дело хозяйское...
http://habrahabr.ru/post/142662/
http://www.kalinin.ru/programming/cpp/31_07_00.shtml
Использование #pragma pack(push,1)... pack(pop)
1
murtukov
8 / 8 / 3
Регистрация: 30.01.2013
Сообщений: 99
06.10.2013, 02:07  [ТС] #14
Пожалуйста, если лень вникать в весь код (а это и не обязательно), то вникните хотя бы в методы saveToFile() и readFromFile()

Добавлено через 1 минуту
alsav22, спасибо за ссылочки, обязательно прочту
0
alsav22
5421 / 4816 / 442
Регистрация: 04.06.2011
Сообщений: 13,587
06.10.2013, 02:09 #15
murtukov, ваш код не компилируется. А у вас? Вы там строки С сравниваете, как string.
0
MoreAnswers
Эксперт
37091 / 29110 / 5898
Регистрация: 17.06.2006
Сообщений: 43,301
06.10.2013, 02:09
Привет! Вот еще темы с ответами:

Си, Запись и Чтение из файла. - C++
Глупый вопрос, но весь гугл перерыл. Нужно сделать именно на СИ(не на с++) чтение и запись в файл .txt . С использованием библиотеки...

Запись и чтение из файла - C++
Здравствуйте, написал программу которая реализует следующее, пользователь создает исходный текстовый файл, в который заносит текст,...

Запись и чтение из файла - C++
Нужно сохранять в файл запись в таком виде Имя Фамилия Телефон Имя ..... помогите плиз реализовать это.И соответсвенно чтение...

Запись и чтение с файла - C++
Есть названия машин, их цена и производитель. Нужно сделать программу, которая при вводе производителя выводит название машины и её цены. ...


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

Или воспользуйтесь поиском по форуму:
15
Yandex
Объявления
06.10.2013, 02:09
Ответ Создать тему
Опции темы

КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin® Version 3.8.9
Copyright ©2000 - 2017, vBulletin Solutions, Inc.
Рейтинг@Mail.ru