gpt4 book ai didi

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

转载 作者:行者123 更新时间:2023-12-01 15:12:55 24 4
gpt4 key购买 nike

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

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

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

我尝试检查 CStdioFile 的源代码iN VC++ 2017 和 2008 一样。

那么是否有不重写整个问题的简单解决方案 CStdioFile类(class)?

最佳答案

CStdioFile主要存在一个原因:它是 constructor采取FILE*争论。此类的目的是包装 C 运行时 file并通过 CFile 公开- 兼容的接口(interface)。该实现继承了所有 C 运行时限制,特别是 2GB 的文件大小限制。
为了解决这个问题,有几种选择:

  • 如果可能,请放弃使用 CStdioFile共。除非您与(遗留)C 代码交互,否则没有明显的理由使用它。
  • 如果这不是一个选项,请从 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)

    关于c++ - CStdioFile 不能处理大于 2GB 的文件?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/62097271/

    24 4 0
    Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
    广告合作:1813099741@qq.com 6ren.com