5

我有一个监视, AND事件WatchService的目录树。问题是 a 的上下文只给出了一个 Path 对象。在删除事件中,我不确定路径是否引用了常规文件的目录。ENTRY_CREATEENTRY_DELETEENTRY-MODIFYWatchEvent<?>

WatchKey key = null;

try {
    key = watcher.take();
} catch (InterruptedException e) {
    e.printStackTrace();
}

for (WatchEvent<?> event : key.pollEvents()) {
    if (event.kind() == StandardWatchEventKinds.OVERFLOW) {
        continue;
    }

    Path parent = (Path) key.watchable();
    Path p = parent.resolve((Path) event.context());

    for (DirectoryModifiedListener listener : listeners) {
        if (event.kind() == StandardWatchEventKinds.ENTRY_DELETE) {
            // only do this if p referred to a file, not a directory
            listener.onFileCreated(p);
        }
    }
}

我的问题是如何确定路径是否p引用了已删除的文件或目录?API公开的内容甚至可能吗?

4

1 回答 1

3

Files.isDirectory(path) 顾名思义是实时检查。

Path 对象(分别是实现 WindowsPath)本身没有状态(或缓存的历史记录)。它只不过是路径的占位符。

因此,可悲的答案是:您无法确定 Path 在删除后是否引用了目录。

一种解决方案是记住 Path 对象的种类,只要它们存在。这意味着在开始观看之前收集一份清单:

Path parent = Paths.get("/my/hotfolder");

Set<String> subdirs = new HashSet<String>();

for (Path p : Files.newDirectoryStream(parent)) {
    if (Files.isDirectory(p))
        subdirs.add(p.getFileName().toString());
}

注册 ENTRY_CREATE 以使列表保持最新

WatchKey key = hotfolder.register(watcher, ENTRY_CREATE, ENTRY_DELETE);

现在您可以确定子对象是否是目录:

for (WatchEvent<?> event : key.pollEvents())
{
    if (event.kind() == StandardWatchEventKinds.OVERFLOW)
        continue;

    Path p = ((Path) key.watchable()).resolve((Path) event.context());

    String filename = p.getFileName().toString();

    if (event.kind() == StandardWatchEventKinds.ENTRY_CREATE && Files.isDirectory(p))
        subdirs.add(filename);

    if (event.kind() == StandardWatchEventKinds.ENTRY_DELETE)
    {
        if (subdirs.contains(filename))
        {
            for (DirectoryModifiedListener listener : listeners)
                listener.onFileDeleted(p);

            subdirs.remove(filename);
        }
    }
}
于 2013-07-06T00:02:35.897 回答