java.nio.file.WatchService 实时监控文件变化的示例代码

作者:志波同学 时间:2021-10-03 01:52:53 

在平时的开发过程中,会有很多场景需要实时监听文件的变化,如下:
1、通过实时监控 mysql 的 binlog 日志实现数据同步
2、修改配置文件后,希望系统可以实时感知
3、应用系统将日志写入文件中,日志监控系统可以实时抓取日志,分析日志内容并进行报警
4、类似 ide 工具,可以实时感知管理的工程下的文件变更

在 Java 语言中,从 JDK7 开始,新增了java.nio.file.WatchService类,用来实时监控文件的变化。

1.示例代码

FileWatchedService 类:

package org.learn.file;

import java.io.IOException;
import java.nio.file.FileSystems;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.nio.file.StandardWatchEventKinds;
import java.nio.file.WatchEvent;
import java.nio.file.WatchKey;
import java.nio.file.WatchService;
import java.util.List;

/**
* 实时监控文件的变化
*
* @author zhibo
* @date 2019-07-30 20:37
*/
public class FileWatchedService {
   private WatchService watchService;

private FileWatchedListener listener;

/**
    *
    * @param path 要监听的目录,注意该 Path 只能是目录,否则会报错 java.nio.file.NotDirectoryException: /Users/zhibo/logs/a.log
    * @param listener 自定义的 listener,用来处理监听到的创建、修改、删除事件
    * @throws IOException
    */
   public FileWatchedService(Path path, FileWatchedListener listener) throws IOException {
       watchService = FileSystems.getDefault().newWatchService();
       path.register(watchService,
               /// 监听文件创建事件
               StandardWatchEventKinds.ENTRY_CREATE,
               /// 监听文件删除事件
               StandardWatchEventKinds.ENTRY_DELETE,
               /// 监听文件修改事件
               StandardWatchEventKinds.ENTRY_MODIFY);
//
//            path.register(watchService,
//                    new WatchEvent.Kind[]{
//                            StandardWatchEventKinds.ENTRY_MODIFY,
//                            StandardWatchEventKinds.ENTRY_CREATE,
//                            StandardWatchEventKinds.ENTRY_DELETE
//                    },
//                    SensitivityWatchEventModifier.HIGH);

this.listener = listener;
   }

private void watch() throws InterruptedException {
       while (true) {
           WatchKey watchKey = watchService.take();
           List<WatchEvent<?>> watchEventList = watchKey.pollEvents();
           for (WatchEvent<?> watchEvent : watchEventList) {
               WatchEvent.Kind kind = watchEvent.kind();

WatchEvent<Path> curEvent = (WatchEvent<Path>) watchEvent;
               if (kind == StandardWatchEventKinds.OVERFLOW) {
                   listener.onOverflowed(curEvent);
                   continue;
               } else if (kind == StandardWatchEventKinds.ENTRY_CREATE) {
                   listener.onCreated(curEvent);
                   continue;
               } else if (kind == StandardWatchEventKinds.ENTRY_MODIFY) {
                   listener.onModified(curEvent);
                   continue;
               } else if (kind == StandardWatchEventKinds.ENTRY_DELETE) {
                   listener.onDeleted(curEvent);
                   continue;
               }
           }

/**
            * WatchKey 有两个状态:
            * {@link sun.nio.fs.AbstractWatchKey.State.READY ready} 就绪状态:表示可以监听事件
            * {@link sun.nio.fs.AbstractWatchKey.State.SIGNALLED signalled} 有信息状态:表示已经监听到事件,不可以接续监听事件
            * 每次处理完事件后,必须调用 reset 方法重置 watchKey 的状态为 ready,否则 watchKey 无法继续监听事件
            */
           if (!watchKey.reset()) {
               break;
           }

}
   }

public static void main(String[] args) {
       try {
           Path path = Paths.get("/Users/zhibo/logs/");
           FileWatchedService fileWatchedService = new FileWatchedService(path, new FileWatchedAdapter());
           fileWatchedService.watch();
       } catch (IOException e) {
           e.printStackTrace();
       } catch (InterruptedException e) {
           e.printStackTrace();
       }
   }
}

FileWatchedListener 类:

package org.learn.file;

import java.nio.file.Path;
import java.nio.file.WatchEvent;

public interface FileWatchedListener {
   void onCreated(WatchEvent<Path> watchEvent);

void onDeleted(WatchEvent<Path> watchEvent);

void onModified(WatchEvent<Path> watchEvent);

void onOverflowed(WatchEvent<Path> watchEvent);
}

FileWatchedAdapter 类:

package org.learn.file;

import java.nio.file.Path;
import java.nio.file.WatchEvent;
import java.text.DateFormat;
import java.text.SimpleDateFormat;
import java.util.Calendar;

/**
* 文件监听适配器
*
* @author zhibo
* @date 2019-07-31 11:07
*/
public class FileWatchedAdapter implements FileWatchedListener {

@Override
   public void onCreated(WatchEvent<Path> watchEvent) {
       Path fileName = watchEvent.context();
       System.out.println(String.format("文件【%s】被创建,时间:%s", fileName, now()));
   }

@Override
   public void onDeleted(WatchEvent<Path> watchEvent) {
       Path fileName = watchEvent.context();
       System.out.println(String.format("文件【%s】被删除,时间:%s", fileName, now()));
   }

@Override
   public void onModified(WatchEvent<Path> watchEvent) {
       Path fileName = watchEvent.context();
       System.out.println(String.format("文件【%s】被修改,时间:%s", fileName, now()));
   }

@Override
   public void onOverflowed(WatchEvent<Path> watchEvent) {
       Path fileName = watchEvent.context();
       System.out.println(String.format("文件【%s】被丢弃,时间:%s", fileName, now()));
   }

private String now(){
       DateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss SSS");
       return dateFormat.format(Calendar.getInstance().getTime());
   }
}

执行以上代码,启动监控任务,然后我在/Users/zhibo/logs/目录中创建、修改、删除文件,命令如下:

java.nio.file.WatchService 实时监控文件变化的示例代码

应用程序感知到文件变化,打印日志如下:

java.nio.file.WatchService 实时监控文件变化的示例代码

2.其实并没有实时

大家可以看到,监控任务基本上是以 10 秒为单位进行日志打印的,也就是说修改一个文件,WatchService 10秒之后才能感知到文件的变化,没有想象中的那么实时。根据以上的经验,推测可能是 WatchService 做了定时的操作,时间间隔为 10 秒。通过翻阅源代码发现,在 PollingWatchService 中确实存在一个固定时间间隔的调度器,如下图:

java.nio.file.WatchService 实时监控文件变化的示例代码

该调度器的时间间隔有 SensitivityWatchEventModifier 进行控制,该类提供了 3 个级别的时间间隔,分别为2秒、10秒、30秒,默认值为 10秒。SensitivityWatchEventModifier 源码如下:

package com.sun.nio.file;

import java.nio.file.WatchEvent.Modifier;

public enum SensitivityWatchEventModifier implements Modifier {
   HIGH(2),
   MEDIUM(10),
   LOW(30);

private final int sensitivity;

public int sensitivityValueInSeconds() {
       return this.sensitivity;
   }

private SensitivityWatchEventModifier(int var3) {
       this.sensitivity = var3;
   }
}

通过改变时间间隔来进行验证,将

path.register(watchService,
               /// 监听文件创建事件
               StandardWatchEventKinds.ENTRY_CREATE,
               /// 监听文件删除事件
               StandardWatchEventKinds.ENTRY_DELETE,
               /// 监听文件修改事件
               StandardWatchEventKinds.ENTRY_MODIFY);

修改为:

path.register(watchService,
                   new WatchEvent.Kind[]{
                           StandardWatchEventKinds.ENTRY_MODIFY,
                           StandardWatchEventKinds.ENTRY_CREATE,
                           StandardWatchEventKinds.ENTRY_DELETE
                   },
                   SensitivityWatchEventModifier.HIGH);

查看日志,发现正如我们的推断,WatchService 正以每 2 秒的时间间隔感知文件变化。
在 stackoverflow 中也有人提出了该问题,问题:Is Java 7 WatchService Slow for Anyone Else,我的 mac 系统中确实存在该问题,由于手头没有 windows、linux 系统,因此无法进行这两个系统的验证。

来源:https://blog.csdn.net/claram/article/details/97919664

标签:java.nio.file.WatchService,监控文件
0
投稿

猜你喜欢

  • Java实现用户管理系统

    2023-08-26 17:18:21
  • 轻松理解Java面试和开发中的IoC(控制反转)

    2023-08-10 03:00:35
  • Java 异步编程实践_动力节点Java学院整理

    2023-06-18 17:04:12
  • Springboot项目快速实现过滤器功能

    2023-08-11 18:30:02
  • java中文传值乱码问题的解决方法

    2023-11-25 16:26:47
  • Java使用JDBC连接Oracle_MSSQL实例代码

    2023-04-19 19:34:46
  • Spring Cloud Ribbon配置详解

    2023-11-25 01:32:50
  • Spring boot中@Conditional和spring boot的自动配置实例详解

    2023-06-20 09:36:14
  • 详解APP微信支付(java后台_统一下单和回调)

    2023-11-10 17:26:42
  • Java使用通配符实现增强泛型详解

    2021-07-06 17:28:01
  • SpringCloud URL重定向及转发代码实例

    2023-11-19 11:43:18
  • Spring Cache手动清理Redis缓存

    2023-11-29 02:49:52
  • Java集合中的fail-fast(快速失败)机制详解

    2023-05-10 16:31:33
  • Spring Boot + Thymeleaf + Activiti 快速开发平台项目 附源码

    2023-11-23 08:23:43
  • Java面试题冲刺第二十五天--并发编程3

    2023-09-11 04:40:10
  • java 动态增加定时任务示例

    2023-07-29 06:56:00
  • springboot项目中使用Swagger的简单示例

    2023-01-14 05:18:24
  • Java删除二叉搜索树的任意元素的方法详解

    2021-10-04 12:27:26
  • 设计模式在Spring框架中的应用汇总

    2023-10-22 19:20:09
  • java开发之内部类的用法

    2023-02-04 21:30:07
  • asp之家 软件编程 m.aspxhome.com