C++ WinApi : ReadDirectoryChangesW() Receiving Double Notifications

标签 c++ windows winapi visual-c++ file-io

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

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

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

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

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

我使用:

  • 非托管 C++
  • Visual Studio 2012
  • Windows 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;
        }
    }

最佳答案

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

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

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

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

关于C++ WinApi : ReadDirectoryChangesW() Receiving Double Notifications,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14036449/

相关文章:

c++ - 虚函数和纯虚函数的区别

javascript - WinJS.UI.Repeater 项目到对象方法的事件处理程序绑定(bind)

c++ - 你怎么知道一个long long的长度

c++ - 使用 ifstream 读取文件时出错

windows - Docker for Windows - 容器域名无法解析

C++:GetPrivateProfileStringA - 随机值?

winapi - Windows服务和Windows驱动程序有什么区别?

windows - DOS MZ 二进制文件可以有 VERSIONINFO 元数据吗?

c++ 如何使两条语句的 sql 与 OLEDB 一起工作?

c++ - 为什么在 Windows 上读取文件到 std::string 的长度错误?