Форум программистов, компьютерный форум, киберфорум
Visual C++
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск Заказать работу  
 
Рейтинг 4.96/26: Рейтинг темы: голосов - 26, средняя оценка - 4.96
87 / 86 / 27
Регистрация: 04.10.2012
Сообщений: 355

Нарушение прав доступа при чтении

15.10.2013, 00:39. Показов 4933. Ответов 5
Метки нет (Все метки)

Студворк — интернет-сервис помощи студентам
Приветствую!
Есть файл, для которого пишу парсер. Структура файла такова: какое-то количество каталогов, каждый из которых хранит какое-то количество записей.
Каждый каталог содержит такие сведения: имя (unsigned long), суммарный размер записей каталога (unsigned long), флаги (unsigned long[4]), записи. Каждая запись выглядит так: имя (unsigned long), длина данных (unsigned short), данные.
Объявил структуру для каталога и записи:
C++
1
2
3
4
5
6
7
8
9
10
11
struct Catalog {
  unsigned long Name, RecordSize, Flags[4];
  void *Records;};
 
struct Record {
  unsigned long Name;
  unsigned short DataSize;
  void *Data;};
 
typedef Catalog *PCatalog;
typedef Record *PRecord;
Так как файл достаточно объемный, парсер читает только инфу о каталогах, а записи парсерятся только при обращении к ним. Класс парсера:
C++
1
2
3
4
5
6
7
8
class Parser {
  char *Buffer;
  unsigned long Count;
  public:
    PCatalog *Items;
    Parser(const TCHAR *FileName);
    virtual ~Parser() {if (Count) {delete[] Buffer; delete[] Items;}}
    virtual unsigned long GetCount() const {return Count;}};
Реализация конструктора:
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
Parser::Parser(const TCHAR *FileName) {
  LARGE_INTEGER fileSize;
  // Открываем файл FileName.
  void *handle = CreateFile(FileName, GENERIC_READ, FILE_SHARE_READ, 0, OPEN_EXISTING, 0, 0);
  if (handle == INVALID_HANDLE_VALUE) throw GetLastError();
  try {
    // Получаем размер файла.
    if (!GetFileSizeEx(handle, &fileSize)) throw GetLastError();
    if (fileSize.HighPart) throw (unsigned long)ERROR_FILE_TOO_LARGE;
    if (fileSize.LowPart) {
      unsigned long size;
      // Заполняем Buffer данными из файла.
      Buffer = new char[fileSize.LowPart];
      if (!readFile(handle, Buffer, fileSize.LowPart, &size, 0)) throw GetLastError();
      if (size != fileSize.LowPart) throw (unsigned long)ERROR_FILE_CORRUPT;
      // Считаем количество каталогов.
      Count = 0;
      for (unsigned long offset = 0; offset < size; offset += *(unsigned long *)&Buffer[offset + 4] + 24) Count++;
      // Заполняем Items указателями на начало каждого каталога в Buffer.
      Items = new PCatalog[Count];
      for (unsigned long index = 0, offset = 0; offset < size; offset += *(unsigned long *)&Buffer[offset + 4] + 24) Items[index++] = (Catalog *)&Buffer[offset];}
    else Count = 0;}
  catch (...) {
    СloseHandle(handle);
    throw;}
  if (!СloseHandle(handle)) throw GetLastError();}
Указатели на каталоги расставляются корректно, могу получить имя любого каталога (Parser->Items[...]->Name).
Теперь класс парсинга записей каталога:
C++
1
2
3
4
5
6
7
class RecordParser {
  unsigned long Count;
  public:
    PRecord *Items;
    RecordParser(const Catalog *Parent);
    virtual ~RecordParser {if (Count) delete[] Items;}
    virtual unsigned long GetCount() const {return Count;}};
Реализация конструктора:
C++
1
2
3
4
5
6
7
8
9
10
RecordParser::RecordParser (const Catalog *Parent) {
  const char *data = (char *)Parent->Records;
  const unsigned long size = Parent->RecordSize;
  // Считаем количество записей.
  Count = 0;
  if (size) {
    for (unsigned long offset = 0; offset < size; offset += *(unsigned short *)&data[offset + 4] + 6) Count++;
    // Заполняем Items указателями на начало каждой записи в data.
    Items = new PRecord[Count];
    for (unsigned long index = 0, offset = 0; offset < size; offset += *(unsigned short *)&data[offset + 4] + 6) Items[index++] = (Record *)&data[offset];}}
Собственно, весь код.
Ошибка всплывает во время работы конструктора RecordParser (строка 7). Происходит "нарушение прав доступа при чтении" data[offset + 4] на первом же витке цикла (точнее, по завершении витка, при обновлении значения offset).
Проверял следующее:
- Файл не поврежден, а его структура корректна.
- Структуры объявлены верно.
- data в конструкторе RecordParser действительно ссылается на начало записей каталога. Следовательно, &data[4] ссылается на размер первой записи (unsigned short).
В чем же дело?

Добавлено через 7 часов 21 минуту
Немного продвинулся в решении проблемы. Понял, что строка
C++
1
const char *data = (char *)Parent->Records;
воспринимается компилятором как
C++
1
const char *data = (char *)(unsigned long)Parent->Records;
То есть data указывает не туда же, куда и Parent->Records, а на адрес, который можно прочесть в (unsigned long)Parent->Records.
Вообще без идей, отчего так.
0
cpp_developer
Эксперт
20123 / 5690 / 1417
Регистрация: 09.04.2010
Сообщений: 22,546
Блог
15.10.2013, 00:39
Ответы с готовыми решениями:

Нарушение прав доступа при чтении
Есть некоторая структура struct Person { char Surname; char Name; sex Sex; int Age; char Group; };

Нарушение прав доступа при чтении
Пишу программу для роботы с базами данных. Скомпилирывал ошибок нема, но при запуске выдает вечно ошибку! вот скрин...

Нарушение прав доступа при чтении в VS10
Доброго времени суток. Наткнулся на проблему, битый час не могу с ней справиться. Имеется файл (любой, но в данном случае простой текст), в...

5
18 / 18 / 4
Регистрация: 05.05.2013
Сообщений: 88
23.10.2013, 01:30
Ну если в data храниться адрес, тогда достаточно просто разименовать его перед использованием:
C++
1
(*data)
0
87 / 86 / 27
Регистрация: 04.10.2012
Сообщений: 355
23.10.2013, 15:47  [ТС]
Нет, в Data хранятся данные. С этой проблемой я уже разобрался. Просто еще раз убедился, что по ночам надо спать, а не работать, и тогда не будут возникать подобные проблемы.
0
1 / 1 / 1
Регистрация: 22.10.2013
Сообщений: 42
10.11.2013, 02:22
не могли бы вы рассказать как справились? у меня такая же ошибка, забрёл сюда из гугла=)
0
0 / 0 / 0
Регистрация: 28.11.2015
Сообщений: 10
27.03.2016, 16:46
Поддерживаю вопрос выше.
0
87 / 86 / 27
Регистрация: 04.10.2012
Сообщений: 355
04.04.2016, 14:55  [ТС]
Ошибка была здесь:
C++
1
2
3
struct Catalog {
  unsigned long Name, RecordSize, Flags[4];
  void *Records;};
По задумке, Records должен был указывать на начало записи. На деле же Records был первыми четырьмя байтами записи (на x86), которые воспринимались как указатель, ссылающийся неизвестно куда.
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
raxper
Эксперт
30234 / 6612 / 1498
Регистрация: 28.12.2010
Сообщений: 21,154
Блог
04.04.2016, 14:55
Помогаю со студенческими работами здесь

Ошибка: нарушение прав доступа при чтении из файла
Реализовал функцию считывания объектов класса из файла. Функция прекрасно считывает из файла, но перед выходом из нее мне выдаёт ошибку:...

Необработанное исключение, нарушение прав доступа при чтении
Необработанное исключение по адресу 0x0F5E4654 (msvcr120d.dll) в Проект1.exe: 0xC0000005: нарушение прав доступа при чтении по адресу...

(C++, asm, winapi) Readfile: Нарушение прав доступа при чтении
Задача: нужно из С++ в ассемблерной вставке обработать текстовый файл, используя функции WinAPI. При вызове ReadFile программа...

Необработанное исключение: 0xC0000005: Нарушение прав доступа при чтении
Программа складывает и вычитает матрицы произвольной размерности, читаемые ею с файла. На самом выходе из программы вылетает данное...

Необработанное исключение в "0x1000bc48" в "*.exe": 0xC0000005: Нарушение прав доступа при чтении
Выдает ошибку после запуска приложения. Указывает на строчку : glutPassiveMotionFunc(MM);. Код: #include &quot;stdafx.h&quot; ...


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

Или воспользуйтесь поиском по форуму:
6
Ответ Создать тему
Новые блоги и статьи
Реалии
Hrethgir 01.03.2026
Нет, я не закончил до сих пор симулятор. Эта задача сложнее. Не получилось уйти в плавсостав, но оно и к лучшему, возможно. Точнее получалось - но сварщиком в палубную команду, а это значит, в моём. . .
Ритм жизни
kumehtar 27.02.2026
Иногда приходится жить в ритме, где дел становится всё больше, а вовлечения в происходящее — всё меньше. Плотный график не даёт вниманию закрепиться ни на одном событии. Утро начинается с быстрых,. . .
SDL3 для Web (WebAssembly): Сборка библиотек: SDL3, Box2D, FreeType, SDL3_ttf, SDL3_mixer и SDL3_image из исходников с помощью CMake и Emscripten
8Observer8 27.02.2026
Недавно вышла версия 3. 4. 2 библиотеки SDL3. На странице официальной релиза доступны исходники, готовые DLL (для x86, x64, arm64), а также библиотеки для разработки под Android, MinGW и Visual Studio. . . .
SDL3 для Web (WebAssembly): Реализация движения на Box2D v3 - трение и коллизии с повёрнутыми стенами
8Observer8 20.02.2026
Содержание блога Box2D позволяет легко создать главного героя, который не проходит сквозь стены и перемещается с заданным трением о препятствия, которые можно располагать под углом, как верхнее. . .
Конвертировать закладки radiotray-ng в m3u-плейлист
damix 19.02.2026
Это можно сделать скриптом для PowerShell. Использование . \СonvertRadiotrayToM3U. ps1 <path_to_bookmarks. json> Рядом с файлом bookmarks. json появится файл bookmarks. m3u с результатом. # Check if. . .
Семь CDC на одном интерфейсе: 5 U[S]ARTов, 1 CAN и 1 SSI
Eddy_Em 18.02.2026
Постепенно допиливаю свою "многоинтерфейсную плату". Выглядит вот так: https:/ / www. cyberforum. ru/ blog_attachment. php?attachmentid=11617&stc=1&d=1771445347 Основана на STM32F303RBT6. На борту пять. . .
Камера Toupcam IUA500KMA
Eddy_Em 12.02.2026
Т. к. у всяких "хикроботов" слишком уж мелкий пиксель, для подсмотра в ESPriF они вообще плохо годятся: уже 14 величину можно рассмотреть еле-еле лишь на экспозициях под 3 секунды (а то и больше),. . .
И ясному Солнцу
zbw 12.02.2026
И ясному Солнцу, и светлой Луне. В мире покоя нет и люди не могут жить в тишине. А жить им немного лет.
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2026, CyberForum.ru