概述
java1.7中 提供了WatchService來監(jiān)控系統中文件的變化州叠。該監(jiān)控是基于操作系統的文件系統監(jiān)控器,可以監(jiān)控系統是所有文件的變化凶赁,這種監(jiān)控是無需遍歷咧栗、無需比較的逆甜,是一種基于信號收發(fā)的監(jiān)控,因此效率一定是最高的致板;現在Java對其進行了包裝交煞,可以直接在Java程序中使用OS的文件系統監(jiān)控器了。
使用場景
- 場景一:比如系統中的配置文件斟或,一般都是系統啟動的時候只加載一次素征,如果想修改配置文件,還須重啟系統萝挤。如果系統想熱加載一般都會定時輪詢對比配置文件是否修改過御毅,如果修改過重新加載。
- 場景二:監(jiān)控磁盤中的文件變化怜珍,一般需要把磁盤中的所有文件全部加載一邊亚享,定期輪詢一遍磁盤,跟上次的文件狀態(tài)對比绘面。如果文件、目錄過多侈沪,每次遍歷時間都很長揭璃,而且還不是實時監(jiān)控。
而以上兩種場景就比較適合使用 WatchService 進行文件監(jiān)控亭罪。
示例
import java.io.IOException;
import java.nio.file.FileSystems;
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;
public class FileWatchServiceDemo {
public static void main(String[] args) throws IOException, InterruptedException {
WatchService watchService = FileSystems.getDefault().newWatchService();
String filePath = "D:/aa";
Paths.get(filePath).register(watchService, StandardWatchEventKinds.ENTRY_CREATE,
StandardWatchEventKinds.ENTRY_MODIFY, StandardWatchEventKinds.ENTRY_DELETE);
while(true){
WatchKey key = watchService.take();
List<WatchEvent<?>> watchEvents = key.pollEvents();
for (WatchEvent<?> event : watchEvents) {
if(StandardWatchEventKinds.ENTRY_CREATE == event.kind()){
System.out.println("創(chuàng)建:[" + filePath + "/" + event.context() + "]");
}
if(StandardWatchEventKinds.ENTRY_MODIFY == event.kind()){
System.out.println("修改:[" + filePath + "/" + event.context() + "]");
}
if(StandardWatchEventKinds.ENTRY_DELETE == event.kind()){
System.out.println("刪除:[" + filePath + "/" + event.context() + "]");
}
}
key.reset();
}
}
}
- 使用 Path 來指定要監(jiān)控的目錄
- Path.register() 方法注冊要監(jiān)控指定目錄的那些事件(創(chuàng)建瘦馍、修改、刪除)
StandardWatchEventKinds.ENTRY_CREATE //創(chuàng)建
StandardWatchEventKinds.ENTRY_MODIFY //修改
StandardWatchEventKinds.ENTRY_DELETE //刪除
- 調用watchService.take(); 獲取監(jiān)控目錄文件的變化的WatchKey应役。該方法是阻塞方法情组,如果沒有文件修改,則一直阻塞箩祥。
- 遍歷所有的修改事件院崇,并做相應處理。
- 完成一次監(jiān)控就需要重置監(jiān)控器袍祖。
不使用 WatchService 監(jiān)控的弊端
- 非常繁瑣底瓣,必須自己手動開啟一個后臺線程每隔一段時間遍歷一次目標節(jié)點并記錄當前狀態(tài),然后和上一次遍歷的狀態(tài)對比蕉陋,如果不相同就表示發(fā)生了變化捐凭,再采取相應的操作,這個過程非常長凳鬓,都需要用戶自己手動實現;
- 效率低:效率都消耗在了遍歷、保存狀態(tài)掘剪、對比狀態(tài)上了诊赊!這是因為舊版本的Java無法很好的利用OS文件系統的功能匹颤,因此只能這樣笨拙地監(jiān)控文件變化;
想了解更多精彩內容請關注我的公眾號