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

时间:2018-11-06 13:30:38

标签: 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

1 个答案:

答案 0 :(得分:2)

我没有安装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

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