java實時監(jiān)控文件行尾內(nèi)容的實現(xiàn)
今天講一下怎樣用Java實現(xiàn)實時的監(jiān)控文件行尾的追加內(nèi)容,類似Linux命令
tail -f
在之前的面試中遇到過一個問題,就是用Java實現(xiàn)tail功能,之前的做法是做一個定時任務(wù)每隔1秒去讀取一次文件,去判斷內(nèi)容是否有追加,如果有則輸出新追加的內(nèi)容,這個做法雖然能勉強實現(xiàn)功能,但是有點太low,今天采用另外一種實現(xiàn)方式,基于事件通知。
1.WatchService
首先介紹一下WatchService類,WatchService可以監(jiān)控某一個目錄下的文件的變動(新增,修改,刪除)并以事件的形式通知文件的變更,這里我們可以實時的獲取到文件的修改事件,然后計算出追加的內(nèi)容,Talk is cheap,Show me the code.
Listener
簡單的接口,只有一個fire方法,當(dāng)事件發(fā)生時處理事件。
public interface Listener {
/**
* 發(fā)生文件變動事件時的處理邏輯
*
* @param event
*/
void fire(FileChangeEvent event);
}
FileChangeListener
Listener接口的實現(xiàn)類,處理文件變更事件。
public class FileChangeListener implements Listener {
/**
* 保存路徑跟文件包裝類的映射
*/
private final Map<String, FileWrapper> map = new ConcurrentHashMap<>();
public void fire(FileChangeEvent event) {
switch (event.getKind().name()) {
case "ENTRY_MODIFY":
// 文件修改事件
modify(event.getPath());
break;
default:
throw new UnsupportedOperationException(
String.format("The kind [%s] is unsupport.", event.getKind().name()));
}
}
private void modify(Path path) {
// 根據(jù)全路徑獲取包裝類對象
FileWrapper wrapper = map.get(path.toString());
if (wrapper == null) {
wrapper = new FileWrapper(path.toFile());
map.put(path.toString(), wrapper);
}
try {
// 讀取追加的內(nèi)容
new ContentReader(wrapper).read();
} catch (IOException e) {
e.printStackTrace();
}
}
}
FileWrapper
文件包裝類,包含文件和當(dāng)前讀取的行號
public class FileWrapper {
/**
* 當(dāng)前文件讀取的行數(shù)
*/
private int currentLine;
/**
* 監(jiān)聽的文件
*/
private final File file;
public FileWrapper(File file) {
this(file, 0);
}
public FileWrapper(File file, int currentLine) {
this.file = file;
this.currentLine = currentLine;
}
public int getCurrentLine() {
return currentLine;
}
public void setCurrentLine(int currentLine) {
this.currentLine = currentLine;
}
public File getFile() {
return file;
}
}
FileChangeEvent
文件變更事件
public class FileChangeEvent {
/**
* 文件全路徑
*/
private final Path path;
/**
* 事件類型
*/
private final WatchEvent.Kind<?> kind;
public FileChangeEvent(Path path, Kind<?> kind) {
this.path = path;
this.kind = kind;
}
public Path getPath() {
return this.path;
}
public WatchEvent.Kind<?> getKind() {
return this.kind;
}
}
ContentReader
內(nèi)容讀取類
public class ContentReader {
private final FileWrapper wrapper;
public ContentReader(FileWrapper wrapper) {
this.wrapper = wrapper;
}
public void read() throws FileNotFoundException, IOException {
try (LineNumberReader lineReader = new LineNumberReader(new FileReader(wrapper.getFile()))) {
List<String> contents = lineReader.lines().collect(Collectors.toList());
if (contents.size() > wrapper.getCurrentLine()) {
for (int i = wrapper.getCurrentLine(); i < contents.size(); i++) {
// 這里只是簡單打印出新加的內(nèi)容到控制臺
System.out.println(contents.get(i));
}
}
// 保存當(dāng)前讀取到的行數(shù)
wrapper.setCurrentLine(contents.size());
}
}
}
DirectoryTargetMonitor
目錄監(jiān)視器,監(jiān)控目錄下文件的變化
public class DirectoryTargetMonitor {
private WatchService watchService;
private final FileChangeListener listener;
private final Path path;
private volatile boolean start = false;
public DirectoryTargetMonitor(final FileChangeListener listener, final String targetPath) {
this(listener, targetPath, "");
}
public DirectoryTargetMonitor(final FileChangeListener listener, final String targetPath, final String... morePaths) {
this.listener = listener;
this.path = Paths.get(targetPath, morePaths);
}
public void startMonitor() throws IOException {
this.watchService = FileSystems.getDefault().newWatchService();
// 注冊變更事件到WatchService
this.path.register(watchService, StandardWatchEventKinds.ENTRY_MODIFY);
this.start = true;
while (start) {
WatchKey watchKey = null;
try {
// 阻塞直到有事件發(fā)生
watchKey = watchService.take();
watchKey.pollEvents().forEach(event -> {
WatchEvent.Kind<?> kind = event.kind();
Path path = (Path) event.context();
Path child = this.path.resolve(path);
listener.fire(new FileChangeEvent(child, kind));
});
} catch (Exception e) {
this.start = false;
} finally {
if (watchKey != null) {
watchKey.reset();
}
}
}
}
public void stopMonitor() throws IOException {
System.out.printf("The directory [%s] monitor will be stop ...\n", path);
Thread.currentThread().interrupt();
this.start = false;
this.watchService.close();
System.out.printf("The directory [%s] monitor will be stop done.\n", path);
}
}
測試類
在D盤新建一個monitor文件夾, 新建一個test.txt文件,然后啟動程序,程序啟動完成后,我們嘗試往test.txt添加內(nèi)容然后保存,控制臺會實時的輸出我們追加的內(nèi)容,PS:追加的內(nèi)容要以新起一行的形式追加,如果只是在原來的尾行追加,本程序不會輸出到控制臺,有興趣的同學(xué)可以擴展一下
public static void main(String[] args) throws IOException {
DirectoryTargetMonitor monitor = new DirectoryTargetMonitor(new FileChangeListener(), "D:\\monitor");
monitor.startMonitor();
}
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
SpringBoot如何在運行時動態(tài)添加數(shù)據(jù)源
這篇文章主要介紹了SpringBoot如何在運行時動態(tài)添加數(shù)據(jù)源,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2019-10-10
使用Idea maven創(chuàng)建Spring項目過程圖解
這篇文章主要介紹了使用Idea maven創(chuàng)建Spring項目過程圖解,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2020-02-02
Jmeter自定義函數(shù)base64加密實現(xiàn)過程解析
這篇文章主要介紹了Jmeter自定義函數(shù)base64加密實現(xiàn)過程解析,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2020-07-07
詳解Springboot Oauth2 Server搭建Oauth2認證服務(wù)
這篇文章主要介紹了Springboot Oauth2 Server 搭建Oauth2認證服務(wù),小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2019-05-05
關(guān)于集合和字符串的互轉(zhuǎn)實現(xiàn)方法
下面小編就為大家?guī)硪黄P(guān)于集合和字符串的互轉(zhuǎn)實現(xiàn)方法。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2016-08-08
SpringBoot中關(guān)于static和templates的注意事項以及webjars的配置
今天小編就為大家分享一篇關(guān)于SpringBoot中關(guān)于static和templates的注意事項以及webjars的配置,小編覺得內(nèi)容挺不錯的,現(xiàn)在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧2019-01-01
零基礎(chǔ)寫Java知乎爬蟲之獲取知乎編輯推薦內(nèi)容
上篇文章我們拿百度首頁做了個小測試,今天我們來個復(fù)雜的,直接抓取知乎編輯推薦的內(nèi)容,小伙伴們可算松了口氣,終于進入正題了,哈哈。2014-11-11

