我正在尝试使用spring执行“WatchService”,但它看起来不可能,因为当我尝试在应用程序上下文时运行此服务但是在控制到来时加载弹簧上下文会停止
key = watcher.take();
由于此应用程序上下文的加载没有发生。
下面是完整代码
@Component
public class DirectoryWatchDemo {
@PostConstruct
public static void test(){
try {
WatchService watcher = FileSystems.getDefault().newWatchService();
Path dir = Paths.get("C:/test");
dir.register(watcher, ENTRY_CREATE);
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 == ENTRY_MODIFY &&
fileName.toString().equals("DirectoryWatchDemo.java")) {
System.out.println("My source file has changed!!!");
}
}
boolean valid = key.reset();
if (!valid) {
break;
}
}
} catch (IOException ex) {
System.err.println(ex);
}
}
}
我这样做是因为我不想手动执行“WatchService”。
答案 0 :(得分:3)
这是因为你有一个无限循环,因此@PostConstruct
的方法调用永远不会返回。
(它对我来说很奇怪,@PostConstruct
可以使用静态方法,但也许这样做有效)
因此,解决方案是为观察者启动一个新线程。你可以用不同的方式做到这一点:
@Async
添加到该方法(我不知道这是否适用于@PostConstruct
方法)(主要缺点是,这是在初始化完整的应用程序上下文之后开始的)@Scheduler
注释:@Scheduled(fixedDelay = Long.MAX_VALUE)
- (优于@PostConstruct
+ @Async
的优势在于:它确实有效,并且在完成上下文初始化之后启动)答案 1 :(得分:1)
WatchService.take等待下一个监视键:“检索并删除下一个监视密钥,如果还没有,则等待。”
PostConstruct - 这是一个Java注释,而不是Spring注释 - “用于在完成依赖注入以执行任何初始化之后需要执行的方法。此方法必须在类之前调用投入使用。“根据这个文档,似乎PostConstruct必须在bean投入使用之前返回(“必须在类投入使用之前调用”)。
但是你的PostConstruct
方法没有返回;所以PostConstruct
不是你需要的。
您可以考虑实现Spring InitializingBean接口,该接口提供回调方法afterPropertiesSet
。它应该允许您启动这种类型的服务方法。
否则,您可以查看Apache VFS2虚拟文件系统,其中包括文件夹观察程序。这就是我的项目所使用的;在系统启动时启动观察器非常容易;此外,它还会监视文件删除,更新和创建事件(与Camel文件观察器不同)。