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()
时,函数将 write/read 字节位于 OVERLAPPED
指定的起始文件偏移量处,然后更新 OVERLAPPED
以包含新文件偏移量 在 字节 [=46] 之后=].您将相同的 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
结构时,函数将从存储在文件句柄本身中的当前文件偏移量开始 write/read 字节,然后更新句柄以存储新的文件偏移 在 字节之后 written/read。因此,如果您使用相同的文件句柄进行写入和读取,则必须使用 SetFilePointer()
or 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 文档以了解有关使用该标志时必须遵循的特定规则的详细信息:
为了进行测试并确保我已完成所有操作以便以后可以开始扩展,我正在尝试创建一个文件句柄,将给定的字节缓冲区写入该文件句柄,然后读取该文件的一部分放入新的测试缓冲区。
我有代码:
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()
时,函数将 write/read 字节位于 OVERLAPPED
指定的起始文件偏移量处,然后更新 OVERLAPPED
以包含新文件偏移量 在 字节 [=46] 之后=].您将相同的 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
结构时,函数将从存储在文件句柄本身中的当前文件偏移量开始 write/read 字节,然后更新句柄以存储新的文件偏移 在 字节之后 written/read。因此,如果您使用相同的文件句柄进行写入和读取,则必须使用 SetFilePointer()
or 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 文档以了解有关使用该标志时必须遵循的特定规则的详细信息: