Createfile2,ReadFile和WriteFile;为什么ReadFile没有读到我在WriteFile中写的内容?

时间:2016-10-25 00:26:58

标签: c++ .net file-io createfile

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

我有代码:

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还是其中的一些组合。非常感谢您的帮助!

编辑:事实证明,ReadFile()返回False,之后的最后一个错误代码是0x57(87) - ERROR_INVALID_PARAMETER。目前正在谷歌上搜索,但如果我无法得到答案,我仍然有兴趣阅读建议。这对我来说并不是很明显我在这里做错了什么,但我的第一个猜测是我在某种程度上错误地使用了positionData。

最终编辑:删除FILE_FLAG_NO_BUFFERING完成了最后的工作。谢谢!

1 个答案:

答案 0 :(得分:1)

当您使用同步文件句柄时将OVERLAPPED结构传递给WriteFile() / ReadFile(),这些函数将在起始文件中写入/读取字节由OVERLAPPED指定的偏移量,然后更新OVERLAPPED以包含写入/读取的字节后的新文件偏移量。您将同一OVERLAPPED传递给WriteFile()ReadFile(),但在将OVERLAPPED传递给ReadFile()之前,您不会倒回ReadFile()的偏移量,因此{ {1}}不读取先前写入的字节。

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文档:

File Buffering