Java监听文件WatchService的简单使用

public interface WatchService
extends Closeable

通过调用其register方法向监视服务注册Watchable对象,返回WatchKey以表示注册。 当检测到对象的事件时,将发信号通知密钥,如果当前未发出信号,则将其排队到监视服务,以便调用poll或take方法的消费者检索密钥并处理事件。 一旦处理了事件,消费者就会调用密钥的reset方法来重置密钥,该密钥允许密钥发出信号并随后用其他事件重新排队。

入门案例:

/**
 * 文件监听
 */
public class WatchServiceDemo implements Runnable{

    public static void main(String[] args) {
        WatchServiceDemo watchServiceDemo = new WatchServiceDemo();
        watchServiceDemo.run();

    }
    @Override
    public void run() {
        String path = "F:\\TestFile";
        WatchService watchService;
            try {
                //WatchService 实例化
                watchService = FileSystems.getDefault().newWatchService();
                //遍历文件
                Files.walkFileTree(Paths.get(path), new SimpleFileVisitor<Path>(){
                    @Override
                    public FileVisitResult preVisitDirectory(Path dir, BasicFileAttributes attrs) throws IOException {
                        //这里监听文件的创建、修改、删除
                        dir.register(watchService,
                                StandardWatchEventKinds.ENTRY_DELETE,
                                StandardWatchEventKinds.ENTRY_CREATE,
                                StandardWatchEventKinds.ENTRY_MODIFY,
                                StandardWatchEventKinds.OVERFLOW);

                        return FileVisitResult.CONTINUE;
                    }
                });

                while (true) {
                WatchKey key = watchService.take();
                for (WatchEvent<?> watchEvent : key.pollEvents()) {
                    WatchEvent.Kind<?> kind = watchEvent.kind();
                    System.out.println("kind=="+kind);
                    if (kind.equals(StandardWatchEventKinds.OVERFLOW)){
                        System.out.println("file overflow...");

                    }
                    if (kind.equals(StandardWatchEventKinds.ENTRY_DELETE)){
                        System.out.println("file delete...");

                    }
                    if (kind.equals(StandardWatchEventKinds.ENTRY_MODIFY)){
                        System.out.println("file modify...");

                    }
                    if (kind.equals(StandardWatchEventKinds.ENTRY_CREATE)){
                        System.out.println("file create...");

                    }

                    WatchEvent<Path> watchEventPath= (WatchEvent<Path>)watchEvent;
                    //获取文件目录
                    Path filePath =  (Path) key.watchable();
                    //获取文件名
                    Path fileName = watchEventPath.context();
                    System.out.println("filePath=="+filePath+" fileName=="+fileName);

                }
                    boolean reset = key.reset();
                    System.out.println("reset==="+reset);

                }

        }catch (Exception e){
                e.printStackTrace();
            }
    }
}

WatchService实例化

WatchService watchService = FileSystems.getDefault().newWatchService();

将Path注册到WatchService中

dir.register(watchService,
           StandardWatchEventKinds.ENTRY_DELETE,
           StandardWatchEventKinds.ENTRY_CREATE,
           StandardWatchEventKinds.ENTRY_MODIFY,
           StandardWatchEventKinds.OVERFLOW);
  • StandardWatchEventKinds.OVERFLOW:事件丢失或失去
  • StandardWatchEventKinds.ENTRY_CREATE:目录内实体创建或本目录重命名
  • StandardWatchEventKinds.ENTRY_MODIFY:目录内实体修改
  • StandardWatchEventKinds.ENTRY_DELETE:目录内实体删除或重命名

WatchKey:该文件节点所绑定的监控信息池,当文件发生变化时所有的变化都会被导入监控池。监控池是静态的,只有当你主动去获取新的监控池时才会有更新的内容加入监控池。这就造成了系统接收到监控信息事件可能稍长的问题。

//获取下一个变化信息的监控池,如果没有变化就一直等待
WatchKey key = watchService.take();
//利用key.pollEvents()方法返回一系列的事件列表 WatchEvent<?> watchEvent
 for (WatchEvent<?> watchEvent : key.pollEvents()) {
 					//得到 监听的事件类型
                    WatchEvent.Kind<?> kind = watchEvent.kind();
                    System.out.println("kind=="+kind);
                    if (kind.equals(StandardWatchEventKinds.OVERFLOW)){
                        System.out.println("file overflow...");

                    }
                }
 WatchEvent<Path> watchEventPath= (WatchEvent<Path>)watchEvent;
                    //获取文件目录
                    Path filePath =  (Path) key.watchable();
                    //获取文件名
                    Path fileName = watchEventPath.context();

每次的到新的事件后,需要重置监听池

key.reset();

key.reset() 方法
每次take()或者poll()操作都会导致线程监控阻塞,每次操作文件可能需要长时间,如果监听目录下有其他事件发生,将会导致事件丢失。重置操作表示重启该线程,后续的事件都会被读到。

key.cancel()方法:关闭监听池服务

  • 2
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
### 回答1: 下面是 Java 代码示例,它使用 `WatchService` 监听文件夹: ``` import java.io.IOException; import java.nio.file.*; public class FileWatcher { public static void main(String[] args) { try { WatchService watcher = FileSystems.getDefault().newWatchService(); Path dir = Paths.get("/path/to/watch"); dir.register(watcher, StandardWatchEventKinds.ENTRY_CREATE, StandardWatchEventKinds.ENTRY_DELETE, StandardWatchEventKinds.ENTRY_MODIFY); System.out.println("Watch Service registered for dir: " + dir.getFileName()); while (true) { WatchKey key; try { key = watcher.take(); } catch (InterruptedException ex) { return; } for (WatchEvent<?> event : key.pollEvents()) { WatchEvent.Kind<?> kind = event.kind(); @SuppressWarnings("unchecked") WatchEvent<Path> ev = (WatchEvent<Path>) event; Path fileName = ev.context(); System.out.println(kind.name() + ": " + fileName); if (kind == StandardWatchEventKinds.ENTRY_MODIFY && fileName.toString().equals("DirectoryToWatch.java")) { System.out.println("My source file has changed!!!"); } } boolean valid = key.reset(); if (!valid) { break; } } } catch (IOException ex) { System.err.println(ex); } } } ``` 该代码创建了一个 `WatchService` 实例,并将目录 `/path/to/watch` 注册为监听的目录。它将监听文件夹内的文件创建、删除和修改事件。 这是一个死循环,监听服务会在目录上等待任何事件。如果事件发生,该代码将打印事件类型(例如 `ENTRY_CREATE`)和事件上下文(即文件名)。如果事件是文件修改事件,且文件名为 `DirectoryToWatch.java`,它将打印一条信息。 请注意,该代码仅提 ### 回答2: Java提供了WatchService类来监听文件夹中的文件变化。下面是一个使用WatchService监听文件夹的示例代码: ```java import java.nio.file.*; import java.nio.file.WatchEvent.Kind; import java.nio.file.attribute.BasicFileAttributes; public class WatchServiceDemo { public static void main(String[] args) throws Exception { // 创建WatchService对象 WatchService watchService = FileSystems.getDefault().newWatchService(); // 需要监听文件夹路径 Path dir = Paths.get("/path/to/directory"); // 注册监听事件,这里我们监听文件的创建、修改和删除事件 dir.register(watchService, StandardWatchEventKinds.ENTRY_CREATE, StandardWatchEventKinds.ENTRY_MODIFY, StandardWatchEventKinds.ENTRY_DELETE); // 启动一个线程来处理文件变化事件 Thread thread = new Thread(() -> { try { while (true) { // 获取下一个文件变化事件 WatchKey key = watchService.take(); // 遍历文件变化事件 for (WatchEvent<?> event : key.pollEvents()) { // 获取事件类型 Kind<?> kind = event.kind(); // 获取事件发生的文件路径 Path file = (Path) event.context(); // 打印事件类型和文件路径 System.out.println(kind + ": " + file); } // 重置WatchKey,否则下一次将无法接收到文件变化事件 key.reset(); } } catch (Exception e) { e.printStackTrace(); } }); // 启动文件监听线程 thread.start(); // 等待线程结束 thread.join(); } } ``` 上述代码中,我们首先创建了一个WatchService对象。然后,通过调用register方法将需要监听文件夹路径和要监听的事件类型注册到WatchService中。在我们的例子中,我们监听文件的创建、修改和删除事件。 接下来,我们创建一个新的线程来处理文件变化事件。在这个线程中,我们通过调用take方法获取下一个文件变化事件的WatchKey对象。然后通过遍历WatchKey的pollEvents方法获取每个事件的类型和文件路径,并输出到控制台。最后,我们重置WatchKey,以便下一次接收文件变化事件。 最后,我们启动文件监听线程,并等待线程结束。 这样,就完成了使用WatchService类来监听文件夹的代码。当文件夹中的文件发生变化时,代码将会输出对应的事件类型和文件路径。 ### 回答3: 使用JavaWatchService类可以方便地在文件夹中进行文件和目录的监视。下面是一个简单Java代码示例,演示如何使用WatchService监听文件夹: ```java import java.io.IOException; import java.nio.file.*; public class FolderWatcher { public static void main(String[] args) { try { // 创建WatchService对象 WatchService watchService = FileSystems.getDefault().newWatchService(); // 注册要监听文件夹 Path folderPath = Paths.get("/path/to/folder"); folderPath.register(watchService, StandardWatchEventKinds.ENTRY_CREATE, StandardWatchEventKinds.ENTRY_DELETE, StandardWatchEventKinds.ENTRY_MODIFY); System.out.println("开始监听文件夹:" + folderPath); // 监听文件夹中的事件 WatchKey key; while ((key = watchService.take()) != null) { for (WatchEvent<?> event : key.pollEvents()) { WatchEvent.Kind<?> kind = event.kind(); if (kind == StandardWatchEventKinds.OVERFLOW) { continue; } // 获取发生的事件和文件路径 WatchEvent<Path> watchEvent = (WatchEvent<Path>) event; Path fileName = watchEvent.context(); // 打印事件类型和文件路径 System.out.println(kind.name() + ": " + folderPath.resolve(fileName)); } // 重置WatchKey对象,以便继续监听 boolean valid = key.reset(); if (!valid) { break; } } } catch (IOException | InterruptedException e) { e.printStackTrace(); } } } ``` 在上面的代码中,首先创建了一个WatchService对象,然后指定要监听文件夹,并将其注册到WatchService中。接着,使用while循环来监听文件夹中的事件。在循环中,使用take()方法从WatchService获取WatchKey对象,并通过pollEvents()方法获取WatchEvent对象列表。在列表中,可以获取到事件的类型和文件路径。最后,使用reset()方法重置WatchKey对象,以便继续监听。 需要注意的是,上述代码中需要替换"/path/to/folder"为实际要监听文件夹路径。另外,监听到的文件和目录变化将会以事件的形式进行处理,可以根据需要进行相应的逻辑操作。 希望以上的回答对您有所帮助!

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值