CStdioFile не может работать с файлами размером более 2 ГБ? - PullRequest
1 голос
/ 30 мая 2020

Я использую Visual C ++ 2008. В VC ++ 2008 CFile поддерживает 2 ^ 64 огромных файла. Поэтому я думаю, что CStdioFile также должен поддерживать.

Однако при использовании CStdioFile::GetLength() для файла размером более 2 ГБ я получаю CFileException, ниже фрагмент кода:

void CTestCStdioFileDlg::OnBnClickedButton1()
{
    // TODO: Add your control notification handler code here
    CStdioFile MyFile;
    CString strLine;
    ULONGLONG uLength;

    strLine = _T("This is a line.");

    if (MyFile.Open(_T("C:\\Temp\\MyTest.dat"), CFile::modeCreate | CFile::modeWrite | CFile::shareExclusive | CFile::typeBinary))
    {
        for (UINT uIndex = 0; uIndex = 200000000; uIndex ++)
        { 
            MyFile.WriteString(strLine);
            uLength = MyFile.GetLength();
        }

        MyFile.Close();
    }
}

После трассировки в CStdio::GetLength() я обнаружил, что следующий фрагмент кода вызовет исключение, как показано ниже:

   nCurrent = ftell(m_pStream);            -> This will return -1
   if (nCurrent == -1)
      AfxThrowFileException(CFileException::invalidFile, _doserrno,
         m_strFileName);

Удивительно, что CStdioFile по-прежнему использует ftell вместо _ftelli64 для работать с потоком.

Затем я ищу в документе CStdioFile, я не могу найти ни одного документа на CStdioFile::GetLength, единственное связанное с ним - https://docs.microsoft.com/en-us/cpp/mfc/reference/cstdiofile-class?view=vs-2019#seek, и он просит меня см. документ fseek. Но в документе fseek я все еще не нахожу ничего, связанного с ограничением размера файла.

Наконец, я нашел сторонний сайт, который указывает, что CStdioFile::GetLength содержит ошибку: http://www.flounder.com/msdn_documentation_errors_and_omissions.htm#CStdioFile :: GetLength , но это не дает решения.

Кроме этого, вряд ли есть какие-либо вопросы или сообщения о 2 ГБ в CStdioFile в Интернете. Это действительно странно.

Я пытаюсь проверить исходный код CStdioFile iN VC ++ 2017, и он такой же, как и для 2008 года.

Итак, есть ли простое решение проблемы без перезаписи всего CStdioFile класс?

1 Ответ

5 голосов
/ 30 мая 2020

CStdioFile существует в основном по одной причине: это конструктор , принимающий аргумент FILE*. Цель этого класса - обернуть C Runtime файл и предоставить его через CFile -совместимый интерфейс. Реализация наследует все C ограничения времени выполнения, в частности ограничение размера файла в 2 ГБ.

Для решения этой проблемы существует несколько вариантов:

  1. Если возможно , полностью отказаться от использования CStdioFile. Если вы не взаимодействуете с (устаревшим) кодом C, нет явных причин для его использования.

  2. Если это не вариант, создайте индивидуальную реализацию из CStdioFile и override все члены класса, которые отображают смещения относительно файлов (GetPosition(), GetLength(), Seek()). Все остальные члены класса не затрагиваются и могут быть просто унаследованы. (Примечание. Обязательно восстановите текущий указатель файла при реализации GetLength().)

    Microsoft предоставляет расширения для своей C среды выполнения, которые предлагают смещения шириной 64 бита ( _ftelli64 и _fseeki64 ).

Если вам нужно go с вариантом 2, следующее расширение CStdioFile можно использовать в качестве замены с поддержкой файлы размером более 2 ГБ.

CStdioFileExt.h:

#pragma once

#include <afx.h>

class CStdioFileExt : public CStdioFile
{
    DECLARE_DYNAMIC(CStdioFileExt)
public:
    ULONGLONG GetPosition() const override;
    ULONGLONG GetLength() const override;
    ULONGLONG Seek(LONGLONG lOff, UINT nFrom) override;
};

CStdioFileExt. cpp:

#include "CStdioFileExt.h"

ULONGLONG CStdioFileExt::GetPosition() const
{
    ASSERT_VALID(this);
    ASSERT(m_pStream != NULL);

    auto const pos = _ftelli64(m_pStream);
    if (pos == -1L)
        AfxThrowFileException(CFileException::invalidFile, _doserrno, m_strFileName);
    return static_cast<ULONGLONG>(pos);
}

ULONGLONG CStdioFileExt::GetLength() const
{
    ASSERT_VALID(this);

    auto const nCurrent = _ftelli64(m_pStream);
    if (nCurrent == -1L)
        AfxThrowFileException(CFileException::invalidFile, _doserrno, m_strFileName);

    auto nResult = _fseeki64(m_pStream, 0, SEEK_END);
    if (nResult != 0)
        AfxThrowFileException(CFileException::badSeek, _doserrno, m_strFileName);

    auto const nLength = _ftelli64(m_pStream);
    if (nLength == -1L)
        AfxThrowFileException(CFileException::invalidFile, _doserrno, m_strFileName);
    nResult = _fseeki64(m_pStream, nCurrent, SEEK_SET);
    if (nResult != 0)
        AfxThrowFileException(CFileException::badSeek, _doserrno, m_strFileName);

    return static_cast<ULONGLONG>(nLength);
}

ULONGLONG CStdioFileExt::Seek(LONGLONG lOff, UINT nFrom)
{
    ASSERT_VALID(this);
    ASSERT(nFrom == begin || nFrom == end || nFrom == current);
    ASSERT(m_pStream != NULL);

    if (_fseeki64(m_pStream, lOff, nFrom) != 0)
        AfxThrowFileException(CFileException::badSeek, _doserrno, m_strFileName);

    auto const pos = _ftelli64(m_pStream);
    return static_cast<ULONGLONG>(pos);
}

IMPLEMENT_DYNAMIC(CStdioFileExt, CStdioFile)
...