创建文件2、读取文件和写入文件;为什么 ReadFile 不读取我在 WriteFile 中编写的内容?

Createfile2, ReadFile, and WriteFile; why isn't ReadFile reading what I've written in WriteFile?

本文关键字:文件 读取 WriteFile ReadFile 创建 为什么      更新时间:2023-10-16

为了测试并确保我已经把所有东西都放下,以便以后可以开始扩展,我试图创建一个文件句柄,将给定的字节缓冲区写入该文件句柄,然后将该文件的一部分读取到新的测试缓冲区中。

我有代码:

const size_t vSize = 0x10000;
std::vector<byte> buffer(vSize, 0);
for (int i = 0; i != vSize; ++i)
{
    buffer[i] = i & 0xff;
}
std::wstring path = ApplicationData::Current->LocalFolder->Path->Data();
std::wstring testFileName = path + std::wstring(L"\TestVariablySized");
_CREATEFILE2_EXTENDED_PARAMETERS extend = { 0 };
extend.dwSize = sizeof(CREATEFILE2_EXTENDED_PARAMETERS);
extend.dwFileAttributes = FILE_ATTRIBUTE_NORMAL;
extend.dwFileFlags = FILE_FLAG_NO_BUFFERING;
extend.dwSecurityQosFlags = SECURITY_ANONYMOUS;
extend.lpSecurityAttributes = nullptr;
extend.hTemplateFile = nullptr;
HANDLE hMappedFile = CreateFile2(
    testFileName.c_str(),
    GENERIC_READ | GENERIC_WRITE,
    0,
    OPEN_ALWAYS,
    &extend);
_OVERLAPPED positionalData;
positionalData.Offset = 0;
positionalData.OffsetHigh = 0;
positionalData.hEvent = 0;
WriteFile(
    hMappedFile,
    &buffer[0],
    vSize,
    NULL,
    positionalData);
std::vector<byte> testBuffer(128);
ReadFile(
    hMappedFile,
    (LPVOID)&testBuffer[0],
    128,
    NULL,
    &positionalData);

不幸的是,当我在之后设置断点并检查testBuffer中的实际内容时,我发现它都是零。我也尝试过在没有positionalData的情况下执行上述所有操作(即在对WriteFile/ReadFile的调用中用NULL替换它(,但这并没有改变结果。类似地,我尝试过用NULL代替CreateFile2的扩展参数,结果也是一样的。

我最终希望能够在给定的文件中选择一个任意的位置来读取字节,所以如果我对positionalData做了一些奇怪的事情,请告诉我。

到目前为止,我不知道问题是在CreateFile2、ReadFile、WriteFile还是它们的组合中。非常感谢您的帮助!

edit:原来ReadFile((返回False,之后的最后一个错误代码是0x57(87(--error_INVALID_PARAMETER。目前我正在谷歌上搜索,但如果我不能完全得到答案,我仍然有兴趣阅读建议。我现在还不清楚我在这里做错了什么,但我的第一个猜测是我在positionalData上犯了一个错误。

最后的编辑:删除FILE_FLAG_NO_BUFFERING完成了最后的工作。非常感谢。

当使用同步文件句柄将OVERLAPPED结构传递给WriteFile()/ReadFile()时,函数将在OVERLAPPED指定的起始文件偏移量处写入/读取字节,然后更新OVERLAPPED,使其包含写入/读取的字节之后的新文件偏移量。您正在将相同的OVERLAPPED传递给WriteFile()ReadFile(),但在将OVERLAPPED的偏移量传递给ReadFile()之前,您没有倒带它,因此ReadFile()不会读取以前写入的字节。

ULARGE_INTEGER ulOffset;
...
ulOffset.QuadPart = 0; // or whatever offset you need
positionalData.Offset = ulOffset.LowPart;
positionalData.OffsetHigh = ulOffset.HighPart;
WriteFile(hMappedFile, ..., &positionalData);
...
ulOffset.QuadPart = 0; // or whatever offset you need
positionalData.Offset = ulOffset.LowPart;
positionalData.OffsetHigh = ulOffset.HighPart;
ReadFile(hMappedFile, ..., &positionalData);

当您传递NULL而不是OVERLAPPED结构时,函数将从存储在文件句柄内的当前文件偏移量开始写入/读取字节,然后更新句柄以在写入/读取的字节之后存储新的文件偏移量。因此,如果您使用相同的文件句柄进行写入和读取,则必须使用SetFilePointer()SetFilePointerEx():倒带文件句柄的当前偏移量

LARGE_INTEGER liOffset;
...
liOffset.QuadPart = 0; // or whatever offset you need
SetFilePointerEx(hMappedFile, liOffset, NULL, FILE_BEGIN);
WriteFile(hMappedFile, ..., NULL);
...
liOffset.QuadPart = 0; // or whatever offset you need
SetFilePointerEx(hMappedFile, liOffset, NULL, FILE_BEGIN);
ReadFile(hMappedFile, ..., NULL);

也就是说,您使用FILE_FLAG_NO_BUFFERING标志打开文件,该标志对文件I/O操作期间的文件偏移量、缓冲区地址和缓冲区大小有非常严格的限制。有关使用该标志时必须遵守的特定规则的详细信息,请阅读MSDN文档:

文件缓冲