概述
java1.7中 提供了WatchService來(lái)監(jiān)控系統(tǒng)中文件的變化。該監(jiān)控是基于操作系統(tǒng)的文件系統(tǒng)監(jiān)控器,可以監(jiān)控系統(tǒng)是所有文件的變化,這種監(jiān)控是無(wú)需遍歷、無(wú)需比較的,是一種基于信號(hào)收發(fā)的監(jiān)控,因此效率一定是最高的;現(xiàn)在Java對(duì)其進(jìn)行了包裝,可以直接在Java程序中使用OS的文件系統(tǒng)監(jiān)控器了。
使用場(chǎng)景
- 場(chǎng)景一:比如系統(tǒng)中的配置文件,一般都是系統(tǒng)啟動(dòng)的時(shí)候只加載一次,如果想修改配置文件,還須重啟系統(tǒng)。如果系統(tǒng)想熱加載一般都會(huì)定時(shí)輪詢對(duì)比配置文件是否修改過(guò),如果修改過(guò)重新加載。
- 場(chǎng)景二:監(jiān)控磁盤中的文件變化,一般需要把磁盤中的所有文件全部加載一邊,定期輪詢一遍磁盤,跟上次的文件狀態(tài)對(duì)比。如果文件、目錄過(guò)多,每次遍歷時(shí)間都很長(zhǎng),而且還不是實(shí)時(shí)監(jiān)控。
而以上兩種場(chǎng)景就比較適合使用 WatchService 進(jìn)行文件監(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 來(lái)指定要監(jiān)控的目錄
- Path.register() 方法注冊(cè)要監(jiān)控指定目錄的那些事件(創(chuàng)建、修改、刪除)
StandardWatchEventKinds.ENTRY_CREATE //創(chuàng)建
StandardWatchEventKinds.ENTRY_MODIFY //修改
StandardWatchEventKinds.ENTRY_DELETE //刪除
- 調(diào)用watchService.take(); 獲取監(jiān)控目錄文件的變化的WatchKey。該方法是阻塞方法,如果沒(méi)有文件修改,則一直阻塞。
- 遍歷所有的修改事件,并做相應(yīng)處理。
- 完成一次監(jiān)控就需要重置監(jiān)控器。
不使用 WatchService 監(jiān)控的弊端
- 非常繁瑣,必須自己手動(dòng)開(kāi)啟一個(gè)后臺(tái)線程每隔一段時(shí)間遍歷一次目標(biāo)節(jié)點(diǎn)并記錄當(dāng)前狀態(tài),然后和上一次遍歷的狀態(tài)對(duì)比,如果不相同就表示發(fā)生了變化,再采取相應(yīng)的操作,這個(gè)過(guò)程非常長(zhǎng),都需要用戶自己手動(dòng)實(shí)現(xiàn);
- 效率低:效率都消耗在了遍歷、保存狀態(tài)、對(duì)比狀態(tài)上了!這是因?yàn)榕f版本的Java無(wú)法很好的利用OS文件系統(tǒng)的功能,因此只能這樣笨拙地監(jiān)控文件變化;
想了解更多精彩內(nèi)容請(qǐng)關(guān)注我的公眾號(hào)
