短版: 修改文件后,我编写/适应监视目录更改的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
答案 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
通过操纵此数据结构,您应该能够从更改中提取行为类型信息