c++ - Createfile2、ReadFile 和 WriteFile;为什么 ReadFile 不读取我在 WriteFile 中写的内容?

标签 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。目前正在谷歌搜索,但如果我不能完全得到答案,我仍然有兴趣阅读建议。我在这里做错了什么对我来说不是很明显,但我的第一个猜测是我以某种方式在 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 文档,了解有关使用该标志时必须遵循的特定规则的详细信息:

File Buffering

关于c++ - Createfile2、ReadFile 和 WriteFile;为什么 ReadFile 不读取我在 WriteFile 中写的内容?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/40229558/

相关文章:

c++ - Vector 的 Fwrite 和 Fread 返回 Segmentation Fault

c++ - 如何在 C++ OpenCv 中使用 vector 点?

.net - 自动化测试

c# - 如何使用 CanContentScroll tr​​ue 获取 ItemsControl 的开发独立像素中的 ScrollViewer 的 VerticalOffset

parsing - 使用 Lua 读取和解析 .MID 文件?

c++ - 通过 Github Actions 构建时如何使用 clang-10 或 gcc-10?

c++ - 列表中的原始指针与唯一指针

c# - 从另一个表获取ID

java - 如何将 DOM 文档写入文件?

mysql - 将sql查询的结果写入mysql中的文件