18

我尝试了解ReadDirectoryChangesW功能,以便我可以有效地了解多个目录中的内容更改(文件被覆盖、文件删除、重命名等)。

我最近的观察之一是,对于每个文件写入操作,我总是收到两个关于单个文件的通知

我非常仔细地追踪了这一点,我敢肯定,如果我覆盖一个文件(比如一个带有新内容的 .txt 文件 - 基本上里面有几个额外的字母),ReadDirectoryChangesW()每次保存该文件都会通知我两次。

这是一件严肃的事情,因为我希望每次更改只收到一次通知。我不希望无意中重复在我的应用程序中应该只发生一次的操作。

这种行为是否已知?请问有没有办法每次更改只收到一个通知?有没有办法有效避免双重通知?

我用:

  • 非托管 C++
  • 视觉工作室 2012
  • 视窗 7 x64

我使用非常基本的代码来做我的测试,但你会想看看它,所以这里是:

HANDLE hDir = CreateFile(
    lpDir,
    FILE_LIST_DIRECTORY,
    FILE_SHARE_WRITE | FILE_SHARE_READ | FILE_SHARE_DELETE,
    NULL, 
    OPEN_EXISTING, 
    FILE_FLAG_BACKUP_SEMANTICS, 
    NULL);

    int nCounter = 0;
    FILE_NOTIFY_INFORMATION strFileNotifyInfo[1024];
    DWORD dwBytesReturned = 0;   

    while(TRUE)
    {
        if( ReadDirectoryChangesW ( hDir, (LPVOID)&strFileNotifyInfo, sizeof(strFileNotifyInfo), FALSE, FILE_NOTIFY_CHANGE_LAST_WRITE, &dwBytesReturned, NULL, NULL) == 0)
        {
            ErrorCheck(_T("Reading Directory Change"));
        }
        else
        {
            _tcout << _T("File Modified: ") << strFileNotifyInfo[0].FileName << endl;
            _tcout << _T("Loop: ") << nCounter++ << endl;
        }
    }
4

2 回答 2

27

ReadDirectoryChangesW() 对文件系统的看法非常短视。它看到文件系统的每一个变化并尽职尽责地报告它们。是的,当您写入文件时,通常会有不止一个。它是您正在使用的特定文件系统的实现细节,但 Windows 中的任何常见文件系统也为存储文件元数据的文件保留目录条目。

因此,您会看到文件数据的写入。但是您也会看到它更改了目录条目。特别是文件大小,可能会在您写入文件并向文件添加数据时发生变化。以及记录在目录条目中的最后写入和最后访问时间戳。否则,api 对所做的更改视而不见,它只看到低级写入。它也完全不知道特定进程要求写入。

这是您必须处理的事情,无法区分这些写入。您所知道的只是“文件已更改”。如何,为什么,由谁以及多久是完全无法发现的。

您必须处理的其他事情是,在生成通知时,写入文件的进程很可能仍然锁定文件。这会阻止您自己对文件做任何有用的事情。像读取文件或复制它很可能会失败。您必须等到该过程完成文件并关闭其对文件的句柄。除了尝试自己打开文件并拒绝任何共享之外,没有办法发现这一点。这需要一个计时器,定期尝试自己获取文件的锁定。一旦你的管道到位,获得多个文件的更改通知就不再重要了。

于 2012-12-26T13:03:40.070 回答
9

这是在记事本中保存文件后与procmon同时记录运行代码的结果操作。有两个来自 ReadDirectoryChangesW() 的通知和两个来自 procmon 的通知。2 IRP_MJ_WRITE 1 来自记事本 (WriteFile) 1 来自系统缓存管理器 (CcWriteBehind) 程序

于 2012-12-26T06:40:32.353 回答