java 监控文件变化 Java使用WatchService监控文件内容变化的示例
Chown 人气:0想了解Java使用WatchService监控文件内容变化的示例的相关内容吗,Chown在本文为您仔细讲解java 监控文件变化的相关知识和一些Code实例,欢迎阅读和指正,我们先划重点:java,监控文件变化,java监控文件内容变化,下面大家一起来学习吧。
场景
系统实现中经常需要能够感知配置文件的变化,然后及时更新上下文。
实现方案
- 自己起一个单独线程,定时加载文件,实现较简单,但是无法保证能够实时捕捉文件变化,同时耗CPU
- 使用commons-io中的 FileAlterationObserver,思想和上面类似,对比前后文件列表的变化,触发对应事件
- JDK 1.7提供的WatchService,利用底层文件系统提供的功能
使用 WatchService
WatchService用来监控一个目录是否发生改变,但是可以通过 WatchEvent 上下文定位具体文件的变化。具体使用过程中要注意以下两点:
- 文件改变可能会触发两次事件(我的理解:文件内容的变更,元数据的变更),可以通过文件的时间戳来控制
- 在文件变化事件发生后,如果立即读取文件,可能所获内容并不完整,建议的做法判断文件的 length > 0
// 监控文件的变化,重新加载 executor.submit(new Runnable() { @Override public void run() { try { final Path path = FileSystems.getDefault().getPath(getMonitorDir()); System.out.println(path); final WatchService watchService = FileSystems.getDefault().newWatchService(); final WatchKey watchKey = path.register(watchService, StandardWatchEventKinds.ENTRY_MODIFY); while (true) { final WatchKey wk = watchService.take(); for (WatchEvent<?> event : wk.pollEvents()) { final Path changed = (Path) event.context(); Path absolute = path.resolve(changed); File configFile = absolute.toFile(); long lastModified = configFile.lastModified(); logger.info(lastModified + "----------------"); // 利用文件时间戳,防止触发两次 if (changed.endsWith(getLicenseName()) && lastModified != LAST_MOD && configFile.length > 0) { logger.info("----------------- reloading -----------------"); LAST_MOD = lastModified; // 保存上一次时间戳 UPDATED = true; // 设置标志位 } } if (UPDATED) { reloadFile(); // 重新加载 } // reset the key boolean valid = wk.reset(); if (!valid) { logger.error("watch key invalid!"); } } } catch (Exception e) { logger.error(""); } } });
参考
加载全部内容