CStdioFile 不能处理大于 2GB 的文件?



我正在使用Visual C++ 2008。在 VC++ 2008 中,CFile支持 2^64 个大文件。所以我认为CStdioFile也应该支持。

但是,在大于 2GB 的文件上使用CStdioFile::GetLength()时,我得到一个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://learn.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,但它没有提供解决方案。

除此之外,几乎没有关于在线CStdioFile2GB限制的问题或帖子。这真的很奇怪。

我尝试检查CStdioFileiN VC++ 2017的源代码,它与2008的源代码相同。

那么,在不重写整个CStdioFile类的情况下,是否有一个简单的问题解决方案呢?

>CStdioFile存在的原因只有一个:它是构造函数接受FILE*参数。此类的目的是包装 C 运行时文件,并通过CFile兼容接口公开该文件。该实现继承了所有 C 运行时限制,特别是 2GB 的文件大小限制。

要解决此问题,有以下几种选择:

  1. 如果可能,请完全放弃使用CStdioFile。除非你与(遗留的(C代码接口,否则没有突出的理由使用它。

  2. 如果这不是一个选项,请从CStdioFile派生一个自定义实现,并override所有显示文件相对偏移量的类成员(GetPosition()GetLength()Seek()(。所有其他类成员不受影响,可以简单地继承。(注意:请确保在实现GetLength()时恢复当前文件指针。

    Microsoft 提供对其 C 运行时的扩展,这些扩展提供 64 位宽偏移量(_ftelli64 和 _fseeki64(。

如果需要使用选项 2,可以使用以下CStdioFile扩展名作为直接替换,支持大于 2GB 的文件。

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)

相关内容

  • 没有找到相关文章

最新更新