Почему фред достигает EOF рано?

Я пишу библиотеку C, которая читает файл в память. Он пропускает первые 54 байта файла (заголовка), а затем читает остаток как данные. Я использую fseek для определения длины файла, а затем использую fread для чтения в файле.

Цикл запускается один раз, а затем заканчивается, потому что достигается EOF (без ошибок). В конце bytesRead = 10624, ftell(stream) = 28726, и буфер содержит 28726 значений. Я ожидаю, что fread будет читать 30000 байт, а позиция файла будет 30054, когда будет достигнут EOF.

Си не мой родной язык, поэтому я подозреваю, что где-то произошла глупая ошибка новичка.

Код выглядит следующим образом:

const size_t headerLen = 54;

FILE * stream;
errno_t ferrno = fopen_s( &stream, filename.c_str(), "r" );
if(ferrno!=0) {
  return -1;
}

fseek( stream, 0L, SEEK_END );
size_t bytesTotal = (size_t)(ftell( stream )) - headerLen; //number of data bytes to read
size_t bytesRead = 0;
BYTE* localBuffer = new BYTE[bytesTotal];
fseek(stream,headerLen,SEEK_SET);
while(!feof(stream) && !ferror(stream)) {
    size_t result = fread(localBuffer+bytesRead,sizeof(BYTE),bytesTotal-bytesRead,stream);
    bytesRead+=result;
}

В зависимости от используемой вами ссылки, вполне очевидно, что добавление "b" к флагу режима является ответом. Поиск кандидатур на нагрудный знак.:-)

Эта ссылка говорит об этом во втором абзаце, втором предложении (хотя и не в их таблице).

MSDN не обсуждает бинарный флаг, пока на полпути вниз страницы.

OpenGroup упоминает о существовании тега "b", но утверждает, что он "не будет иметь никакого эффекта".

3 ответа

Решение

Возможно, это проблема двоичного режима. Попробуйте открыть файл с "r+b" как режим.

РЕДАКТИРОВАТЬ: как отмечено в комментарии "rb" вероятно, лучше соответствует вашему первоначальному замыслу, так как "r+b" откроет его для чтения / записи и "rb" только для чтения.

Также стоит отметить, что простое включение binmode.obj в вашу команду ссылки сделает это за вас при открытии всех файлов.

Решение, основанное на предыдущих ответах:

    size_t bytesRead = 0;
    BYTE* localBuffer = new BYTE[bytesTotal];
    fseek(stream,headerLen,SEEK_SET);
        while(!feof(stream) && !ferror(stream)) {
        size_t result = fread(localBuffer+bytesRead,sizeof(BYTE),bytesTotal-
        bytesRead,stream);
    bytesRead+=result;
}
Другие вопросы по тегам