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

Чтение двоичного файла в объект класса - C++

Восстановить пароль Регистрация
 
skafiend
0 / 0 / 0
Регистрация: 13.03.2010
Сообщений: 10
10.04.2011, 12:59     Чтение двоичного файла в объект класса #1
Здравствуйте.
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
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
#include <iostream>
#include <fstream>
 
/* strlcpy based on OpenBSDs strlcpy */
#include <stdio.h>
#include <sys/types.h>
 
size_t strlcpy(char *, const char *, size_t);
 
// Размер имени
unsigned int nSizeName=25;
// Размер фамилии
unsigned int nSizeFirstName=25;
// Размер рабочего телефона
unsigned int nSizePhone=11;
// Размер мобильного телефона
unsigned int nSizeMobilePhone=12;
 
using namespace std;
 
// Класс описывающий телефонный справочник из четырех полей
class PhoneBook
{
  public:
    PhoneBook();
    ~PhoneBook();
    void EditData();
    char* GetData(char nChoise);
//  private:
    char* pszName;
    char* pszFirstName;
    char* pszPhone;
    char* pszMobilePhone;
};
 
//  Конструктор выделяющий место под переменные в куче и вызавающий функцию
//+ EditData() для их инициализации
PhoneBook::PhoneBook()
{
  this->pszName = new char[nSizeName];
  this->pszFirstName = new char[nSizeFirstName];
  this->pszPhone = new char[nSizePhone];
  this->pszMobilePhone = new char [nSizeMobilePhone];
 
  PhoneBook::EditData();
};
 
//  Деструктор, уничтожающий переменный после завершения программы т.е. освобож
//+ ющий место в ОЗУ
PhoneBook::~PhoneBook()
{
  delete this->pszName;
  delete this->pszFirstName;
  delete this->pszPhone;
  delete this->pszMobilePhone;
};
 
//  Функция осуществляющая редактирование одной записи. Не проверяет введенные 
//+ символы, а только длину занесенных переменных посредством strlcpy
void PhoneBook::EditData()
{
  char* szTemp = new char[255];
 
  cout << "\nВведите Имя: "; cin >> szTemp;
  strlcpy(this->pszName, szTemp, nSizeName);
  cout << "Введите Фамилию: "; cin >> szTemp;
  strlcpy(this->pszFirstName, szTemp, nSizeFirstName);
  cout << "Введите Рабочий Телефон: "; cin >> szTemp;
  strlcpy(this->pszPhone, szTemp, nSizePhone);
  cout << "Введите Мобильный Телефон: "; cin >> szTemp;
  strlcpy(this->pszMobilePhone, szTemp, nSizeMobilePhone);
};
 
//  Принимает в качестве аргумента символ, который определяет возвращаемое значе
//+ ние. N - имя, F - фамилия, P - рабочий телефон, M - мобильный
char* PhoneBook::GetData(char cChoise)
{
    if (cChoise == 'N')
    {
      return pszName;
    }
      else
      {
         if (cChoise == 'F')
         {
           return pszFirstName;
         }
           else
           {
              if (cChoise == 'P')
              {
                 return pszPhone;
              }
                else
                {
                   if (cChoise == 'M')
                   {
                      return pszMobilePhone;
                   }
                }
           }
      }
};
 
main()
{
  // Количество записей в справочнике
  unsigned int nMax = 0;
 
  cout << "-----------------------------------------\n";
  cout << "Введите число записей: "; cin >> nMax;
  cout << "-----------------------------------------";
  //  Если удалось привести введенное значение к типу int, то продолжим выполне
  //+ ние, иначе выходим
  if (int(nMax))
  {
  // Объявляем массив объектов, конструктор автоматически вызывается для каждого
  PhoneBook Book[nMax];
  // Регистрирует выход из цикла
  bool exit;
 
  // Пока exit не стал true, предлагаем действия с книгой
  while (!exit)
  {
  // Регистрирует присутствие результатов в поиске
  bool result = false;
  // Счетчик в циклах
  unsigned int nCount = 0;
  // Выбор числовой
  unsigned int nChoise = 0;
  // Выбор символьный
  char cChoise = NULL;
  // Фраза для поиска
  string szPhrase;
 
    cout << "-----------------------------------------\n";
    cout << "Что Вы будете делать дальше?\n[1]Просмотр всех записей\n[2]Редактирование записи\n[3]Поиск записи\n[4]Записать результат в файл *.txt\n[5]Сохранить результаты в двоичный файл\n[6]Прочитать из двоичного файла\n[7]Выход\n";
    cout << "-----------------------------------------\n";
 
    //  Будет очищен входной поток и проигнорировано 10000 неверных символов, т.е. неверная обработка возможна, но
    //+ очень хочется посмотреть на пользователя, который попробует их набрать, зацикливания нет в любом случае
    cout << "Ваш выбор: "; cin >> nChoise; cin.clear(); cin.ignore(10000, '\n');
    cout << "-----------------------------------------\n";
 
  // Экземпляр класса ofstream
  ofstream fPhoneBook;
  // Экземпляр класса fstream
  fstream BinaryBook;
 
  // Строка
    switch ((int)nChoise)
    {
      case 1:
              // Перебор всех записей
              while (nCount < nMax)
              {
                cout << "Номер записи: " << nCount << "\n";
                cout << "Имя: "; cout << Book[nCount].PhoneBook::GetData('N') << "\n";
                cout << "Фамилия: "; cout << Book[nCount].PhoneBook::GetData('F') << "\n";
                cout << "Рабочий: "; cout << Book[nCount].PhoneBook::GetData('P') << "\n";
                cout << "Мобильный: "; cout << Book[nCount].PhoneBook::GetData('M') << "\n";
                nCount++;
              }
              break;
      case 2:
              cout << "Нумерация начинается с нуля!\n";
              cout << "Номер записи для редактирования: "; cin >> nChoise;
              cout << "-----------------------------------------";
              //  Если введенное значение больше, чем индекс последнего элемента
              //+ вывести сообщение об ошибке
              if (nChoise > nMax - 1)
              {
                cout << "Вы ввели неверный номер!\n"; break;
              }
                else
                {
                  // Иначе вызвать функцию редактирования
                  Book[nChoise].PhoneBook::EditData();
                };
              break;
      case 3:
              // Бесконечный цикл, который прекратит крутиться при верном вводе
              while (true)
              {
                cout << "По какому параметру поиск:\n[N]По имени\n[F]По фамилии\n[P]По рабочему\n[M]По мобильному\n";
                cout << "-----------------------------------------\n" << "Ваш выбор: ";
                cin >> cChoise; cin.clear(); cin.ignore(10000,'\n');
                if (!(cChoise == 'N' || cChoise == 'F' || cChoise == 'P' || cChoise == 'M'))
                {
                  cout << "-----------------------------------------\n";
                  cout << "Неверный ввод!\n";
                  cout << "-----------------------------------------\n";
                }
                else 
                {
                  break;
                }
              }
              cout << "-----------------------------------------\n";
              cout << "Введите фразу для точного поиска!\n" << "Фраза: ";
              cin >> szPhrase;
              cout << "-----------------------------------------\n";
              // Перебор всех записей
              while (nCount < nMax)
              {
                 // Нашли точное соответствие? Вывести его!
                 if (Book[nCount].PhoneBook::GetData(cChoise) == szPhrase)
                 {
                    cout << "Номер записи: " << nCount << "\n";
                    cout << "Имя: "; cout << Book[nCount].PhoneBook::GetData('N') << "\n";
                    cout << "Фамилия: "; cout << Book[nCount].PhoneBook::GetData('F') << "\n";
                    cout << "Рабочий: "; cout << Book[nCount].PhoneBook::GetData('P') << "\n";
                    cout << "Мобильный: "; cout << Book[nCount].PhoneBook::GetData('M') << "\n";
                    result = true;
                 };
                 nCount++;
              };
              // Значение result не изменилось.
              if (result == false) { cout << "Ничего не найдено!\n";}
              break;
      case 4:
              fPhoneBook.open("PhoneBook.txt");
              if (fPhoneBook != NULL)
              {
                 nCount = 0;
                 while (nCount < nMax)
                 {
                   fPhoneBook << "-----------------------------------------" << '\n'
                              << "Номер записи: " << nCount << '\n'
                              << "Имя: " << Book[nCount].PhoneBook::GetData('N') << '\n'
                              << "Фамилия: " << Book[nCount].PhoneBook::GetData('F') << '\n'
                              << "Рабочий: " << Book[nCount].PhoneBook::GetData('P') << '\n'
                              << "Мобильный: " << Book[nCount].PhoneBook::GetData('M') << '\n';
 
                   nCount++;
                 }
                 fPhoneBook << "-----------------------------------------" << '\n';
                 cout << "Открыт на запись успешно!\n";
              }
              else
              {
                cout << "Файл открыть не удалось!\n";
              };
              fPhoneBook.close(); 
              break;
      case 5:
              // Открываем бинарный файл на запись
              BinaryBook.open("MyBook", ios::out | ios::binary | ios::trunc);
 
              if(!BinaryBook)
              {
                cout << "He удается открыть файл.\n";
                return 1;
              }
                else
                {
                  cout << "Открыт на запись успешно!\n";
                }
 
              nCount = 0;
              while(nCount < nMax)
              {
                BinaryBook.write((char *) &Book[nCount] , sizeof Book[nCount] ) ; // Записываем блок данных.
                nCount++;
              }
 
              BinaryBook.close ();
              break;
      case 6:
              // Открываем бинарный файл на чтение
              BinaryBook.open("MyBook", ios::in | ios::binary);
 
              if(!BinaryBook)
              {
                cout << "He удается открыть файл.\n";
                return 1;
              }
                else
                {
                  cout << "Открыт на чтение успешно!\n";
                }
 
              nCount = 0;
              while(!BinaryBook.eof())
              {
                BinaryBook.read((char *) &Book[nCount] , sizeof Book[nCount] ) ; // Считываем блок данных.
                nCount++;
              }
 
              BinaryBook.close ();
              break;
      case 7: exit=true; break; // Выход из цикла
      default: cout << "Ваш ввод неверен! Повторите снова!\n"; break;
    }
  }
 
  }
  else
  {
    cout << "\nВаш ввод неверен!";
    cout << "\n-----------------------------------------\n";
  }
}
 
 /*
 * Copy src to string dst of size siz.  At most siz-1 characters
 * will be copied.  Always NUL terminates (unless siz == 0).
 * Returns strlen(src); if retval >= siz, truncation occurred.
 */
size_t
strlcpy(char *dst, const char *src, size_t siz)
{
        char *d = dst;
        const char *s = src;
        size_t n = siz;
 
        /* Copy as many bytes as will fit */
        if (n != 0 && --n != 0) {
                do {
                        if ((*d++ = *s++) == 0)
                                break;
                } while (--n != 0);
        }
 
        /* Not enough room in dst, add NUL and traverse rest of src */
        if (n == 0) {
                if (siz != 0)
                        *d = '\0';                /* NUL-terminate dst */
                while (*s++)
                        ;
        }
 
        return(s - src - 1);        /* count does not include NUL */
}
Не получается разобраться, что не так с 269 строкой. В файл все пишется а вот обратно возвращаться не хочет. Спасибо за помощь.
Similar
Эксперт
41792 / 34177 / 6122
Регистрация: 12.04.2006
Сообщений: 57,940
10.04.2011, 12:59     Чтение двоичного файла в объект класса
Посмотрите здесь:

C++ чтение двоичного файла
C++ Вывод в файл и последующее чтение из файла объектов класса
Сохранение и чтение объекта класса в файл/из файла C++
Можно ли считывать поля класса из двоичного файла одним fread C++
Удалить записанный экземпляр класса из двоичного файла C++
C++ Перегрузка операторов. Запись и чтение из файла обьект класса
Указатель на объект базового класса и адрес объекта производного класса C++
C++ Не могу считать из файла. Объект класса Medicament в текстовом файле

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

Или воспользуйтесь поиском по форуму:
После регистрации реклама в сообщениях будет скрыта и будут доступны все возможности форума.
Ответ Создать тему
Опции темы

Текущее время: 09:27. Часовой пояс GMT +3.
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin® Version 3.8.9
Copyright ©2000 - 2016, vBulletin Solutions, Inc.
Рейтинг@Mail.ru