等待文件可用于使用Win32进行读取
我正在通过同步调用ReadDirectoryChangesW
观看目录。 当一个新文件可用时,我尝试使用GENERIC_READ
和FILE_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文件。 或者感知创建它们的程序的状态。