CStdioFile не может работать с файлами размером более 2 ГБ?
Я использую 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, и он просит меня увидетьfseek
документ. Но вfseek
документ, я до сих пор не нашел ничего, связанного с ограничением размера файла.
Наконец я нахожу сторонний сайт, который указывает CStdioFile::GetLength
содержит ошибку: http://www.flounder.com/msdn_documentation_errors_and_omissions.htm, но не предлагает решения.
Кроме этого, вряд ли есть какие-либо вопросы или сообщения о лимите 2 ГБ CStdioFile
онлайн. Это действительно странно.
Я пытаюсь проверить исходный код CStdioFile
iN VC++ 2017 и то же, что и в 2008 году.
Так есть ли простое решение проблемы без переписывания всего CStdioFile
класс?
1 ответ
CStdioFile существует в основном по одной причине: его конструктор принимаетFILE*
аргумент. Цель этого класса - обернуть файл среды выполнения C и предоставить его черезCFile
-совместимый интерфейс. Реализация наследует все ограничения среды выполнения C, в частности ограничение размера файла в 2 ГБ.
Чтобы решить эту проблему, есть несколько вариантов:
Если возможно, откажитесь от использования
CStdioFile
все вместе. Если вы не взаимодействуете с (устаревшим) кодом C, нет явной причины когда-либо его использовать.Если это не вариант, создайте собственную реализацию из
CStdioFile
а такжеoverride
все члены класса, которые отображают смещения относительно файлов (GetPosition()
,GetLength()
,Seek()
). Все остальные члены класса не затронуты и могут быть просто унаследованы. (Примечание. Обязательно восстановите текущий указатель файла при реализацииGetLength()
.)Microsoft предоставляет расширения для своей среды выполнения C, которые предлагают 64-битные смещения ( _ftelli64 и _fseeki64).
Если вам нужно выбрать вариант 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)