Java NIO異步文件通道原理及用法解析
在Java 7,AsynchronousFileChannel 被添加到了Java NIO中。使用AsynchronousFileChannel可以實(shí)現(xiàn)異步地讀取和寫(xiě)入文件數(shù)據(jù)。
創(chuàng)建一個(gè)AsynchronousFileChannel
我們可以使用AsynchronousFileChannel提供的靜態(tài)方法 open() 創(chuàng)建它。示例代碼如下:
Path path = Paths.get("data/test.xml");
AsynchronousFileChannel fileChannel =
AsynchronousFileChannel.open(path, StandardOpenOption.READ);
第一個(gè)參數(shù)是一個(gè) PATH 的對(duì)像實(shí)例,它指向了那個(gè)與 AsynchronousFileChannel 相關(guān)聯(lián)的文件。
第二個(gè)參數(shù)是一個(gè)或多個(gè)操作選項(xiàng),它決定了 AsynchronousFileChannel 將對(duì)目標(biāo)文件做何種操作。示例代碼中我們使用了 StandardOpenOption.READ ,它表明我們將要對(duì)目標(biāo)文件進(jìn)行讀操作。
讀取數(shù)據(jù)
AsynchronousFileChannel 提供了兩種讀取數(shù)據(jù)的方式,都是調(diào)用它本身的 read() 方法。下面將對(duì)兩種方式進(jìn)行介紹。
使用Futrue讀取數(shù)據(jù)
第一種反式是調(diào)用 AsynchronousFileChannel 的 read() 方法,該方法反回一個(gè) Future 類(lèi)型的對(duì)象。
Future operation = fileChannelread(buffer, 0);
第一個(gè)參數(shù)是ByteBuffer,從 AsynchronousFileChannel 中讀取的數(shù)據(jù)先寫(xiě)入這個(gè) ByteBuffer 。
第二個(gè)參數(shù)表示從文件讀取數(shù)據(jù)的開(kāi)始位置。
此 read() 方法會(huì)立即返回,即使整個(gè)讀的過(guò)程還沒(méi)有完全結(jié)束。我們可以通過(guò)operation.isDone()來(lái)檢查讀取是否完成。這里的 operation 是上面調(diào)用 read() 方法返回的 Future 類(lèi)型的實(shí)例。下面是一段詳細(xì)的代碼示例:
AsynchronousFileChannel fileChannel = AsynchronousFileChannel.open(path, StandardOpenOption.READ); ByteBuffer buffer = ByteBuffer.allocate(1024); long position = 0; Future<Integer> operation = fileChannel.read(buffer, position); while(!operation.isDone()); buffer.flip(); byte[] data = new byte[buffer.limit()]; buffer.get(data); System.out.println(new String(data)); buffer.clear();
上面的程序首先創(chuàng)建了一個(gè) AsynchronousFileChannel 對(duì)象,然后調(diào)用它的read()方法返回一個(gè)Future。其中read()方法需要兩個(gè)參數(shù),一個(gè)是ByteBuffer,另一個(gè)是讀取文件的開(kāi)始位置。然后通過(guò)循環(huán)調(diào)用isDone() 方法檢測(cè)讀取過(guò)程是否完成,完成后 isDone()方法將返回true。盡管這樣讓cpu空轉(zhuǎn)了一會(huì),但是我們還是應(yīng)該等讀取操作完成后再進(jìn)行后續(xù)的步驟。
一旦讀取完成,數(shù)據(jù)被存儲(chǔ)到ByteBuffer,然后將數(shù)據(jù)轉(zhuǎn)化為字符串既而輸出。
使用CompletionHandler讀取數(shù)據(jù)
第二種讀取數(shù)據(jù)的方式是調(diào)用AsynchronousFileChannel 的另一個(gè)重載 read() 方法,改方法需要一個(gè)CompletionHandler 作為參數(shù)。下面是代碼示例:
fileChannel.read(buffer, position, buffer, new CompletionHandler<Integer, ByteBuffer>() {
@Override
public void completed(Integer result, ByteBuffer attachment) {
System.out.println("result = " + result);
attachment.flip();
byte[] data = new byte[attachment.limit()];
attachment.get(data);
System.out.println(new String(data));
attachment.clear();
}
@Override
public void failed(Throwable exc, ByteBuffer attachment) {
}
});
一旦讀取操作完成,CompletionHandler的 complete() 方法將會(huì)被調(diào)用。它的第一個(gè)參數(shù)是個(gè) Integer類(lèi)型,表示讀取的字節(jié)數(shù)。第二個(gè)參數(shù) attachment 是 ByteBuffer 類(lèi)型的,用來(lái)存儲(chǔ)讀取的數(shù)據(jù)。它其實(shí)就是由 read() 方法的第三個(gè)參數(shù)。當(dāng)前示例中,我們選用 ByteBuffer 來(lái)存儲(chǔ)數(shù)據(jù),其實(shí)我們也可以選用其他的類(lèi)型。
讀取失敗的時(shí)候,CompletionHandler的 failed()方法會(huì)被調(diào)用。
寫(xiě)入數(shù)據(jù)
就像讀取一樣,我們同樣有兩種方式向 AsynchronousFileChannel 寫(xiě)入數(shù)據(jù)。我們可以調(diào)用它的2個(gè)重載的 write() 方法。下面我們將分別加以介紹。
使用Future讀取數(shù)據(jù)
AsynchronousFileChannel也可以異步寫(xiě)入數(shù)據(jù)。下面是一個(gè)完整的寫(xiě)入示例:
AsynchronousFileChannel也可以異步寫(xiě)入數(shù)據(jù)。下面是一個(gè)完整的寫(xiě)入示例:
Path path = Paths.get("data/test-write.txt");
AsynchronousFileChannel fileChannel =
AsynchronousFileChannel.open(path, StandardOpenOption.WRITE);
ByteBuffer buffer = ByteBuffer.allocate(1024);
long position = 0;
buffer.put("test data".getBytes());
buffer.flip();
Future<Integer> operation = fileChannel.write(buffer, position);
buffer.clear();
while(!operation.isDone());
System.out.println("Write done");
首先實(shí)例化一個(gè)寫(xiě)入模式的 AsynchronousFileChannel, 然后創(chuàng)建一個(gè) ByteBuffer 并寫(xiě)入一些數(shù)據(jù)。再然后將數(shù)據(jù)寫(xiě)入文件。最后,檢查返回的 Future,看是否寫(xiě)入完成。
注意,寫(xiě)入目標(biāo)文件要提前創(chuàng)建好,如果它不存在的話,writh() 方法會(huì)拋出一個(gè) java.nio.file.NoSuchFileException。
我們可以用以下方式來(lái)解決這一問(wèn)題:
if(!Files.exists(path)){
Files.createFile(path);
}
使用CompletionHandler寫(xiě)入數(shù)據(jù)
我們也可以使用 CompletionHandler代替Future向AsynchronousFileChannel寫(xiě)入數(shù)據(jù),這種方式可以更加直接的知道寫(xiě)入過(guò)程是否完成。下面是示例程序:
Path path = Paths.get("data/test-write.txt");
if(!Files.exists(path)){
Files.createFile(path);
}
AsynchronousFileChannel fileChannel =
AsynchronousFileChannel.open(path, StandardOpenOption.WRITE);
ByteBuffer buffer = ByteBuffer.allocate(1024);
long position = 0;
buffer.put("test data".getBytes());
buffer.flip();
fileChannel.write(buffer, position, buffer, new CompletionHandler<Integer, ByteBuffer>() {
@Override
public void completed(Integer result, ByteBuffer attachment) {
System.out.println("bytes written: " + result);
}
@Override
public void failed(Throwable exc, ByteBuffer attachment) {
System.out.println("Write failed");
exc.printStackTrace();
}
});
當(dāng)寫(xiě)入程序完成時(shí),CompletionHandler的completed()方法將會(huì)被調(diào)用,相反的如果寫(xiě)入失敗則會(huì)調(diào)用failed()方法。
要留意CompletionHandler的方法的參數(shù) attachemnt是怎么使用的。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- java 基礎(chǔ)知識(shí)之網(wǎng)絡(luò)通信(TCP通信、UDP通信、多播以及NIO)總結(jié)
- Java Socket編程實(shí)例(四)- NIO TCP實(shí)踐
- Java NIO 文件通道 FileChannel 用法及原理
- docker版es、milvus、minio啟動(dòng)命令詳解
- Java BIO,NIO,AIO總結(jié)
- 詳解java NIO之Channel(通道)
- 詳細(xì)了解JAVA NIO之Buffer(緩沖區(qū))
- 簡(jiǎn)單了解JAVA NIO
- 基于Java寫(xiě)minio客戶(hù)端實(shí)現(xiàn)上傳下載文件
- 淺析NIO系列之TCP
相關(guān)文章
關(guān)于maven install報(bào)錯(cuò)原因揭秘:parent.relativePath指向錯(cuò)誤的本地POM文件
在使用Maven進(jìn)行項(xiàng)目構(gòu)建時(shí),如果遇到'parent.relativePath'指向錯(cuò)誤的本地POM文件的問(wèn)題,可能會(huì)導(dǎo)致構(gòu)建失敗,這通常是由于父項(xiàng)目POM文件的相對(duì)路徑設(shè)置錯(cuò)誤、本地POM文件與父項(xiàng)目POM文件版本或內(nèi)容不一致所致,解決方法包括檢查并修正父項(xiàng)目POM文件中的相對(duì)路徑設(shè)置2024-09-09
jxl 導(dǎo)出數(shù)據(jù)到excel的實(shí)例講解
下面小編就為大家分享一篇jxl 導(dǎo)出數(shù)據(jù)到excel的實(shí)例講解,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2017-12-12
Java實(shí)現(xiàn)的圖片高質(zhì)量縮放類(lèi)定義與用法示例
這篇文章主要介紹了Java實(shí)現(xiàn)的圖片高質(zhì)量縮放類(lèi)定義與用法,涉及java針對(duì)圖片的運(yùn)算與轉(zhuǎn)換等相關(guān)操作技巧,需要的朋友可以參考下2017-11-11
簡(jiǎn)單理解java泛型的本質(zhì)(非類(lèi)型擦除)
泛型在java中有很重要的地位,在面向?qū)ο缶幊碳案鞣N設(shè)計(jì)模式中有非常廣泛的應(yīng)用。泛型是參數(shù)化類(lèi)型的應(yīng)用,操作的數(shù)據(jù)類(lèi)型不限定于特定類(lèi)型,可以根據(jù)實(shí)際需要設(shè)置不同的數(shù)據(jù)類(lèi)型,以實(shí)現(xiàn)代碼復(fù)用。下面小編來(lái)簡(jiǎn)單講一講泛型2019-05-05
簡(jiǎn)單總結(jié)SpringMVC攔截器的使用方法
今天給大家?guī)?lái)的是關(guān)于SpringMVC攔截器的相關(guān)知識(shí),文章圍繞著SpringMVC攔截器的使用方法展開(kāi),文中有非常詳細(xì)的介紹及代碼示例,需要的朋友可以參考下2021-06-06
基于Java SSM實(shí)現(xiàn)Excel數(shù)據(jù)批量導(dǎo)入
這篇文章主要為大家詳細(xì)介紹了基于Java SSM如何實(shí)現(xiàn)excel數(shù)據(jù)批量導(dǎo)入,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-11-11
Java 把json對(duì)象轉(zhuǎn)成map鍵值對(duì)的方法
這篇文章主要介紹了java 把json對(duì)象中轉(zhuǎn)成map鍵值對(duì)的方法,本文的目的是把json串轉(zhuǎn)成map鍵值對(duì)存儲(chǔ),而且只存儲(chǔ)葉節(jié)點(diǎn)的數(shù)據(jù) 。需要的朋友可以參考下2018-04-04

