小师妹学JavaIO之:文件系统和WatchService

简介: 小师妹学JavaIO之:文件系统和WatchService

目录



简介



小师妹这次遇到了监控文件变化的问题,F师兄给小师妹介绍了JDK7 nio中引入的WatchService,没想到又顺道普及了一下文件系统的概念,万万没想到。


监控的痛点



小师妹:F师兄最近你有没有感觉到呼吸有点困难,后领有点凉飕飕的,说话有点不顺畅的那种?


没有啊小师妹,你是不是秋衣穿反了?


小师妹:不是的F师兄,我讲的是心里的感觉,那种莫须有的压力,还有一丝悸动缠绕在心。


别绕弯子了小师妹,是不是又遇到问题了。


更多精彩内容且看:



更多内容请访问www.flydean.com


小师妹:还是F师兄懂我,这不上次的Properties文件用得非常上手,每次修改Properties文件都要重启java应用程序,真的是很痛苦。有没有什么其他的办法呢?


办法当然有,最基础的办法就是开一个线程定时去监控属性文件的最后修改时间,如果修改了就重新加载,这样不就行了。


小师妹:写线程啊,这么麻烦,有没有什么更简单的办法呢?


就知道你要这样问,还好我准备的比较充分,今天给你介绍一个JDK7在nio中引入的类WatchService。


WatchService和文件系统



WatchService是JDK7在nio中引入的接口:


image.png


监控的服务叫做WatchService,被监控的对象叫做Watchable:


WatchKey register(WatchService watcher,
                      WatchEvent.Kind<?>[] events,
                      WatchEvent.Modifier... modifiers)
        throws IOException;
WatchKey register(WatchService watcher, WatchEvent.Kind<?>... events)
        throws IOException;


Watchable通过register将该对象的WatchEvent注册到WatchService上。从此只要有WatchEvent发生在Watchable对象上,就会通知WatchService。


WatchEvent有四种类型:


  1. ENTRY_CREATE 目标被创建
  2. ENTRY_DELETE 目标被删除
  3. ENTRY_MODIFY 目标被修改
  4. OVERFLOW 一个特殊的Event,表示Event被放弃或者丢失


register返回的WatchKey就是监听到的WatchEvent的集合。


现在来看WatchService的4个方法:


  1. close 关闭watchService
  2. poll 获取下一个watchKey,如果没有则返回null
  3. 带时间参数的poll 在等待的一定时间内获取下一个watchKey
  4. take 获取下一个watchKey,如果没有则一直等待


小师妹:F师兄,那怎么才能构建一个WatchService呢?


上次文章中说的文件系统,小师妹还记得吧,FileSystem中就有一个获取WatchService的方法:


public abstract WatchService newWatchService() throws IOException;


我们看下FileSystem的结构图:


image.png


在我的mac系统上,FileSystem可以分为三大类,UnixFileSystem,JrtFileSystem和ZipFileSystem。我猜在windows上面应该还有对应的windows相关的文件系统。小师妹你要是有兴趣可以去看一下。


小师妹:UnixFileSystem用来处理Unix下面的文件,ZipFileSystem用来处理zip文件。那JrtFileSystem是用来做什么的?


哎呀,这就又要扯远了,为什么每次问问题都要扯到天边....


从前当JDK还是9的时候,做了一个非常大的改动叫做模块化JPMS(Java Platform Module System),这个Jrt就是为了给模块化系统用的,我们来举个例子:


public void useJRTFileSystem(){
        String resource = "java/lang/Object.class";
        URL url = ClassLoader.getSystemResource(resource);
        log.info("{}",url);
    }


上面一段代码我们获取到了Object这个class的url,我们看下如果是在JDK8中,输出是什么:


jar:file:/Library/Java/JavaVirtualMachines/jdk1.8.0_171.jdk/Contents/Home/jre/lib/rt.jar!/java/lang/Object.class


输出结果是jar:file表示这个Object class是放在jar文件中的,后面是jar文件的路径。


如果是在JDK9之后:


jrt:/java.base/java/lang/Object.class


结果是jrt开头的,java.base是模块的名字,后面是Object的路径。看起来是不是比传统的jar路径更加简洁明了。


有了文件系统,我们就可以在获取系统默认的文件系统的同时,获取到相应的WatchService:


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


WatchSerice的使用和实现本质



小师妹:F师兄,WatchSerice是咋实现的呀?这么神奇,为我们省了这么多工作。


其实JDK提供了这么多类的目的就是为了不让我们重复造轮子,之前跟你讲监控文件的最简单办法就是开一个独立的线程来监控文件变化吗?其实.....WatchService就是这样做的!


PollingWatchService() {
        // TBD: Make the number of threads configurable
        scheduledExecutor = Executors
            .newSingleThreadScheduledExecutor(new ThreadFactory() {
                 @Override
                 public Thread newThread(Runnable r) {
                     Thread t = new Thread(null, r, "FileSystemWatcher", 0, false);
                     t.setDaemon(true);
                     return t;
                 }});
    }


上面的方法就是生成WatchService的方法,小师妹看到没有,它的本质就是开启了一个daemon的线程,用来接收监控任务。


下面看下怎么把一个文件注册到WatchService上面:


private void startWatcher(String dirPath, String file) throws IOException {
        WatchService watchService = FileSystems.getDefault().newWatchService();
        Path path = Paths.get(dirPath);
        path.register(watchService, ENTRY_MODIFY);
        Runtime.getRuntime().addShutdownHook(new Thread(() -> {
            try {
                watchService.close();
            } catch (IOException e) {
                log.error(e.getMessage());
            }
        }));
        WatchKey key = null;
        while (true) {
            try {
                key = watchService.take();
                for (WatchEvent<?> event : key.pollEvents()) {
                    if (event.context().toString().equals(fileName)) {
                        loadConfig(dirPath + file);
                    }
                }
                boolean reset = key.reset();
                if (!reset) {
                    log.info("该文件无法重置");
                    break;
                }
            } catch (Exception e) {
                log.error(e.getMessage());
            }
        }
    }


上面的关键方法就是path.register,其中Path是一个Watchable对象。


然后使用watchService.take来获取生成的WatchEvent,最后根据WatchEvent来处理文件。


相关文章
使用pip时报错:No module named ‘chardet‘ 的解决办法
使用pip时报错:No module named ‘chardet‘ 的解决办法
2325 0
使用pip时报错:No module named ‘chardet‘ 的解决办法
|
消息中间件 数据采集 JSON
iLogtail社区版使用入门 - iLogtail日志处理实战
本文将会详细介绍使用iLogtail社区版进行日志采集时,对于常见日志格式的处理实战。为了便于调测,以下样例默认将采集到的日志,经过处理后,输出到标准输出,开发者可以根据实际需求进行适当的调整。
3399 0
|
SQL Java 数据库连接
挺详细的spring+springmvc+mybatis配置整合|含源代码
挺详细的spring+springmvc+mybatis配置整合|含源代码
|
2月前
|
Shell 测试技术 API
Claude Code 官方内部团队最佳实践!
Immerse,独立开发者、内容创作者、AGI实践者,分享编程、AI、开源等内容。关注公众号“沉浸式趣谈”及个人网站获取更新。欢迎点赞、评论、转发支持!本文介绍Claude Code——智能编程命令行工具及其使用技巧。
1001 0
|
11月前
|
存储 自然语言处理 算法
“无”中生有:基于知识增强的RAG优化实践
本文作者基于自身在RAG技术领域长达半年的实践经验,分享了从初识RAG的潜力到面对实际应用挑战的心路历程,以及如何通过一系列优化措施逐步解决这些挑战的过程。
950 20
“无”中生有:基于知识增强的RAG优化实践
|
11月前
|
存储 缓存 Linux
【Linux】另一种基于rpm安装yum的方式
通过本文的方法,您可以在离线环境中使用RPM包安装YUM并进行必要的配置。这种方法适用于无法直接访问互联网的服务器或需要严格控制软件源的环境。通过配置本地YUM仓库,确保了软件包的安装和更新可以顺利进行。希望本文能够为您在特定环境中部署YUM提供实用的指导。
968 0
|
存储 分布式计算 Hadoop
Hadoop节点HDFS元数据与数据块的关系
【5月更文挑战第19天】
417 4
|
Web App开发 存储 缓存
离线网络搜索
离线网络搜索是指在本地计算机或移动设备上进行网络搜索,而不是通过互联网连接到远程服务器进行搜索。这种技术可以用于在没有网络连接或网络连接不稳定的情况下进行搜索,或者出于隐私或安全考虑而需要保护搜索历史记录和搜索活动。
758 3
|
机器学习/深度学习 算法
【MATLAB】GA_BP神经网络回归预测算法
【MATLAB】GA_BP神经网络回归预测算法
347 3
【MATLAB】GA_BP神经网络回归预测算法