等待文件可用于使用Win32进行读取

我正在通过同步调用ReadDirectoryChangesW观看目录。 当一个新文件可用时,我尝试使用GENERIC_READFILE_SHARE_READ CreateFile立即访问它,但这给了我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 (...); 

我知道,关闭文件上的通知没有用户模式API。 你提出的循环可能是最好的方法。 你可以做的唯一其他事情是在filter驱动程序ala Process Monitor中观察CloseFile,但是…

我认为没有关于你正在寻找的事件的通知,但作为改进,我建议逐步延迟。 这样,如果用户在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 } 

正如@Matt Davis所说,遗憾的是没有用户模式的API,但有一种解决方法,根据你的用例(我在下面写的)可能会做你想要的。

在调用ReadDirectoryChangesW时,过去对我FILE_NOTIFY_CHANGE_LAST_WRITE是注册FILE_NOTIFY_CHANGE_LAST_WRITE 而不是 FILE_NOTIFY_CHANGE_FILE_NAME

 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模式。

如果您对文件的创建方式有所了解,可能要等到文件停止增长X秒,或者等到删除了一个sentinel文件。 或者感知创建它们的程序的状态。