9

我正在通过ReadDirectoryChangesW同步调用来查看目录。CreateFile当有新文件可用时,我尝试使用GENERIC_READand立即访问它FILE_SHARE_READ,但这给了我ERROR_SHARING_VIOLATION. 在我尝试读取文件时,将文件放入监视目录的过程尚未完成写入。

有没有办法可靠地等待文件可供读取?我可以将该方法放入如下所示的循环中,但我希望有更好的方法。

while ((hFile = CreateFile (path, GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING, 0, NULL)) == INVALID_HANDLE_VALUE)
{
    if (GetLastError() == ERROR_SHARING_VIOLATION)
        Sleep (500);
    else
        break; // some other error occurred
}

if (hFile == INVALID_HANDLE_VALUE)
{
    // deal with other error
    return 0;
}

ReadFile (...);
4

4 回答 4

8

我不认为有你正在寻找的那种事件的通知,但作为一种改进,我建议逐步延迟。这样,如果用户在 Excel 中保持文件打开一个小时,您将获得快速响应时间,例如拖放,并且不会以紧密的循环占用 CPU。

int delay= 10;
while ((hFile = CreateFile (path, GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING, 0, NULL)) == INVALID_HANDLE_VALUE)
{
    if (GetLastError() == ERROR_SHARING_VIOLATION) {
        Sleep (delay);
        if (delay<5120) // max delay approx 5.Sec
            delay*= 2;
    }
    else
        break; // some other error occurred
}
于 2009-11-17T18:55:40.853 回答
2

我知道,对于已关闭文件的通知,没有用户模式 ​​API。您提出的循环确实可能是最好的方法。您唯一能做的就是CloseFile在过滤器驱动程序ala Process Monitor中观察,但是糟糕......

于 2009-11-17T05:39:23.627 回答
2

正如@Matt Davis 所说,不幸的是没有用户模式API,但是有一种解决方法,取决于您的用例(我在下面写了我的)可能会满足您的需求。

过去对我有用的是注册FILE_NOTIFY_CHANGE_LAST_WRITE 而不是 FILE_NOTIFY_CHANGE_FILE_NAME打电话时ReadDirectoryChangesW

ZeroMemory(&overlapped, sizeof(OVERLAPPED));
overlapped.hEvent = hChangeEvent;

// ...    
ReadDirectoryChangesW(hSpoolPath,
                      eventBuffer,
                      EVENT_BUF_LENGTH,
                      FALSE,
                      FILE_NOTIFY_CHANGE_LAST_WRITE, // <----
                      NULL,
                      &overlapped,
                      NULL);
// ...
HANDLE events[2];

events[0] = hChangeEvent;
events[1] = hCancelEvent;

DWORD wRc = WaitForMultipleObjects(2, events, FALSE, DIRECTORY_WATCH_TIMEOUT);

一旦拥有进程在创建文件并写入文件后关闭句柄,最后写入时间就会更新。

我的用例是一个通过 TCP/IP 接收 HTTP 请求并将 HTTP 正文写入目录的进程,一旦接收进程完成写入(并因此关闭句柄),另一个进程就会将其拾取。http-server 是写入该目录的唯一进程,因此我可以依赖 create-write-close 模式。

于 2014-10-13T06:52:58.967 回答
0

如果你知道文件是如何创建的,也许等到文件停止增长 X 秒,或者等到一个哨兵文件被删除。或者感知创建它们的程序的状态。

于 2009-11-17T05:41:36.383 回答