繁体   English   中英

Createfile2,ReadFile和WriteFile; 为什么ReadFile不读取我在WriteFile中编写的内容?

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

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

我有代码:

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() SetFilePointer()退文件句柄的当前偏移量:

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文档以获取有关使用该标志时必须遵循的特定规则的详细信息:

文件缓冲

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM