2

下面的代码有两个问题。首先,我一直在搜索这个论坛和其他各种论坛,以寻找我的 1784 错误代码的答案,但我尝试过的一切都失败了。我在 stackoverflow 上检查的两个线程是WriteFile 返回错误 1784BlockWrite I/O Error 1784。我已经检查了这个论坛上的其他一些人,但我不记得现在具体是什么了。

我正在尝试将结构数组保存到一个空的二进制文件中。第一个问题是,如果我的大小变量(nNumberOfBytesToWrite 参数)小于 99000 字节,我就会遇到访问冲突。这个数字跳来跳去。有一段时间,当我测试它时,如果它是 99,999 字节而不是 100,000 字节,则会出现访问冲突。当然,我最终想要做的是将大小设置为整个数组的大小。现在要处理的原始代码已被注释掉,因此我可以使用各种尺寸进行测试。

发生的第二件事(如果我没有遇到访问冲突)是我收到错误代码 1784 并且 WriteFile 每次都失败。正如该主题的其他线程所述,这在 MSDN 上被定义为 ERROR_INVALID_USER_BUFFER 并且描述是“提供的用户缓冲区对于请求的操作无效”。我查看了 MSDN 自己的示例来打开这样的文件(http://msdn.microsoft.com/en-us/library/windows/desktop/bb540534%28v=vs.85%29.aspx)并尝试了一些基于他们的代码的变化,但似乎没有任何效果。

这个问题可能是大量的菜鸟,我确信我忽略了一些简单得可笑的事情,但如果有人有建议,他们将不胜感激。

case IDM_SAVE1:
{
    HANDLE hFile = CreateFile("MineSave.mss", GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);
    int test_buffer[] = {1,2,3,4,5,6,7,8,9,10};

    if(hFile != INVALID_HANDLE_VALUE)
    {
        BOOL bSuccess;
        DWORD size = 100000; //DWORD size = (((sizeof(tile)) * tiles_total));
        LPDWORD bytes_written = 0;
        bSuccess = WriteFile(hFile, test_buffer, size, bytes_written, NULL);
        if(bSuccess)
        {
            MessageBox(hwnd, "File saved successfully.", "Great Job!", MB_OK);
        }
        else
        {
            DWORD error = GetLastError();
            MessageBox(hwnd, "Could not write to file.", "Error", MB_OK);
        }

        CloseHandle(hFile);
    }
    else
    {
        MessageBox(hwnd, "Could not create file.", "Error", MB_OK);
    }
}
break;
4

2 回答 2

7

您的缓冲区大小为 10 个整数,在 Windows 上为 40 个字节。您正在尝试从该缓冲区写入 100,000 个字节。那是未定义的行为,缓冲区溢出。因此访问冲突。

您不能将大于 的值传递sizeof(test_buffer)40nNumberOfBytesToWrite参数WriteFile

您需要循环写入此文件,一次写入 40 个字节,直到您写入所需的内容为止。也许是这样的:

BOOL bSuccess = TRUE;
DWORD bytesRemaining = 100000;
while (bSuccess && bytesRemaining>0)
{
    DWORD bytesToWrite = std::min(sizeof(test_buffer), bytesRemaining);
    DWORD bytesWritten;
    bSuccess = WriteFile(hFile, test_buffer, bytesToWrite, &bytesWritten, NULL);
    bytesRemaining -= bytesToWrite;
}
if (!bSuccess)
{
    //handle error;
}

一次写入 40 个字节非常慢。您会发现每次调用WriteFile.

请注意,如果您也传递给,则不允许传递NULL给参数,就像您在此处所做的那样。从文档中:lpNumberOfBytesWrittenNULLlpOverlapped

lpNumberOfBytesWritten [输出,可选]

……

只有当 lpOverlapped 参数不为 NULL 时,此参数才可以为 NULL。

于 2012-09-28T16:29:34.810 回答
1

必须提供一个缓冲区来接收写入的字节数,lpNumberOfBytesWritten参数必须为非 NULL,或者lpOverlapped参数必须为非 NULL。

您为两者都传递了 NULL,这是非法的并导致访问冲突。

于 2012-09-28T16:59:23.780 回答