用于监视目录的 Python 脚本在文件更新时会触发两次。为什么?

标签 python readdirectorychangesw

简短版本: 我编写/改编的一个Python脚本用于监视目录的更改,当文件被修改时会触发两次。为什么?

长版:

我正在编写一些 Python 代码来监视目录及其子目录的更改。

我从 http://timgolden.me.uk/python/win32_how_do_i/watch_directory_for_changes.html 的“使用 ReadDirectoryChanges API”部分中找到的示例开始

(出于公司 IT 原因,使用 Python Watchdog 包不适合我。)

从示例中剪切并粘贴:

import os

import win32file
import win32con

ACTIONS = {
  1 : "Created",
  2 : "Deleted",
  3 : "Updated",
  4 : "Renamed from something",
  5 : "Renamed to something"
}
# Thanks to Claudio Grondi for the correct set of numbers
FILE_LIST_DIRECTORY = 0x0001

path_to_watch = "."
hDir = win32file.CreateFile (
  path_to_watch,
  FILE_LIST_DIRECTORY,
  win32con.FILE_SHARE_READ | win32con.FILE_SHARE_WRITE | win32con.FILE_SHARE_DELETE,
  None,
  win32con.OPEN_EXISTING,
  win32con.FILE_FLAG_BACKUP_SEMANTICS,
  None
)
while 1:
  #
  # ReadDirectoryChangesW takes a previously-created
  # handle to a directory, a buffer size for results,
  # a flag to indicate whether to watch subtrees and
  # a filter of what changes to notify.
  #
  # NB Tim Juchcinski reports that he needed to up
  # the buffer size to be sure of picking up all
  # events when a large number of files were
  # deleted at once.
  #
  results = win32file.ReadDirectoryChangesW (
    hDir,
    1024,
    True,
    win32con.FILE_NOTIFY_CHANGE_FILE_NAME |
     win32con.FILE_NOTIFY_CHANGE_DIR_NAME |
     win32con.FILE_NOTIFY_CHANGE_ATTRIBUTES |
     win32con.FILE_NOTIFY_CHANGE_SIZE |
     win32con.FILE_NOTIFY_CHANGE_LAST_WRITE |
     win32con.FILE_NOTIFY_CHANGE_SECURITY,
    None,
    None
  )
  for action, file in results:
    full_filename = os.path.join (path_to_watch, file)
    print full_filename, ACTIONS.get (action, "Unknown")

总的来说,这个例子运行良好并且达到了我想要的效果。特别是在创建文件时它工作得很好。但是,当编辑/修改/更新文件时,末尾的打印语句(代表我真正想做的操作)会触发两次。

为什么会发生这种情况?如何预防它,或者至少解决它?我最好的想法是设置一个标志,第一次为 True,第二次为 False。然而,这感觉就像是一个拼凑。

关于可能相关的问题,在哪里可以找到有关 win32con 包和 Microsoft ReadDirectoryChanges API 的文档?我已经进行了一些谷歌搜索,但没有找到任何我认为有用的东西。

哦,是的 - 我在 Windows 7 Enterprise 上运行 Python 3.5.1。

编辑: 好吧,看来我所看到的可能是 ReadDirectoryChangesW() 所固有的。我发现这个 StackOverflow 线程似乎基本上是相同的问题,只是原始海报使用的是 C++,而不是 Python。 C++ WinApi: ReadDirectoryChangesW() Receiving Double Notifications

最佳答案

我没有安装 python 2.7,所以我无法通过查看这部分来自行测试

 win32con.FILE_NOTIFY_CHANGE_FILE_NAME |
 win32con.FILE_NOTIFY_CHANGE_DIR_NAME |
 win32con.FILE_NOTIFY_CHANGE_ATTRIBUTES |
 win32con.FILE_NOTIFY_CHANGE_SIZE | #this will change when someone writes or deletes to the file
 win32con.FILE_NOTIFY_CHANGE_LAST_WRITE |#this will change when someone modifies the file
 win32con.FILE_NOTIFY_CHANGE_SECURITY,

看看同一个操作如何触发这两个条件?这可能是触发您行为的原因,请尝试删除两者之一

编辑以澄清以下评论:

为了了解每种类型的更改的行为,请尝试以下操作:

results = {}
results['FName_Change'] = win32file.ReadDirectoryChangesW (
                        hDir,
                        1024,
                        True,
                        win32con.FILE_NOTIFY_CHANGE_FILE_NAME,
                        None,
                        None
                      )
results['DName_Change'] = win32file.ReadDirectoryChangesW (
                        hDir,
                        1024,
                        True,
                         win32con.FILE_NOTIFY_CHANGE_DIR_NAME, 
                        None,
                        None
                        )
  results['Attributes_Change'] = win32file.ReadDirectoryChangesW (
                        hDir,
                        1024,
                        True,
                         win32con.FILE_NOTIFY_CHANGE_ATTRIBUTES, 
                        None,
                        None

通过操作此数据结构,您应该能够从更改中提取行为类型信息

关于用于监视目录的 Python 脚本在文件更新时会触发两次。为什么?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/53172955/

相关文章:

python - 在 Python 中如何将 `email.message.Message` 对象转换为 `email.message.EmailMessage` 对象

c++ - 我可以使用 FindFirstChangeNotification 和 FindNextChangeNofication 而不是 ReadDirectoryChangesW 获取指定目录中的文件信息更改吗?

filesystemwatcher - 在 DataOntap 上使用 FileSystemWatcher

c - 使用 ReadDirectoryChangesW API 监视目录

python - 如何对二进制字符串列表进行排序

python - OneClassSVM scikit 学习

python - 部分函数与 lambda 的映射列表

python matplotlib 给出了错误的边界角