Createfile2、ReadFile 和 WriteFile;为啥 ReadFile 不读取我在 WriteFile 中写的内容?

Posted

技术标签:

【中文标题】Createfile2、ReadFile 和 WriteFile;为啥 ReadFile 不读取我在 WriteFile 中写的内容?【英文标题】:Createfile2, ReadFile, and WriteFile; why isn't ReadFile reading what I've written in WriteFile?Createfile2、ReadFile 和 WriteFile;为什么 ReadFile 不读取我在 WriteFile 中写的内容? 【发布时间】:2016-10-25 00:26:58 【问题描述】:

为了测试并确保我已经完成所有操作以便以后可以开始扩展,我正在尝试创建一个文件句柄,将给定的字节缓冲区写入该文件句柄,然后读取部分将该文件放入新的测试缓冲区中。

我有代码:

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 在最后完成了这项工作。谢谢!

【问题讨论】:

ReadFile() 返回一个指示成功或错误的 BOOL。检查是否为 FALSE,如果是,请致电GetLastError() 查找错误代码。 感谢您的提示。 ReadFile() 确实返回 false,它给了我错误代码 0x57 (87)。现在在谷歌上搜索。 错误 87 是 ERROR_INVALID_PARAMETERReadFile() 失败,因为您没有关注 restrictions imposed by the FILE_FLAG_NO_BUFFERING flag。如果您摆脱该标志,那么代码可能会起作用。否则,请更新代码以正确遵守规则。 你是对的,删除 FILE_FLAG_NO_BUFFERING 使其工作。 【参考方案1】:

当您将OVERLAPPED 结构传递给WriteFile()/ReadFile() 并使用同步 文件句柄时,函数将在@987654327 指定的起始文件偏移处写入/读取字节@,然后更新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

【讨论】:

教育,谢谢。但是,我遇到的问题仍然存在,即使在写入后将 positionalData.Offset 重置为 0。 将你的标记为正确,因为这至少是让它工作的一半。另一半是您在顶部留下的最后一条评论。编辑:现在这是您编辑后的完整答案。

以上是关于Createfile2、ReadFile 和 WriteFile;为啥 ReadFile 不读取我在 WriteFile 中写的内容?的主要内容,如果未能解决你的问题,请参考以下文章

串口:ReadFile 和 CloseHandle [重复]

51nod——1548 欧姆诺姆和糖果

readfile() 和 fopen() 的区别

readFile() 和 readFileSync() 之间的区别

使用 ReadFile 和 WriteFile 时出现死锁

使用 ReadFile 时访问冲突读取位置