淺析Java?NIO?直接緩沖區(qū)和非直接緩沖區(qū)
定義

以上是書《深入理解java虛擬機》對直接內(nèi)存的描述。直接緩沖區(qū)用的就是直接內(nèi)存。
- java nio字節(jié)緩沖區(qū)要么是直接的,要么是非直接的。如果為直接字節(jié)緩沖區(qū),則java虛擬機會盡最大努力直接在此緩沖區(qū)上執(zhí)行本機的IO操作,也就是說,在每次調(diào)用基礎(chǔ)操作系統(tǒng)的一個本機IO操作前后,虛擬機都會盡量避免將內(nèi)核緩沖區(qū)內(nèi)容復(fù)制到用戶進程緩沖區(qū)中,或者反過來,盡量避免從用戶進程緩沖區(qū)復(fù)制到內(nèi)核緩沖區(qū)中。
- 直接緩沖區(qū)可以通過調(diào)用該緩沖區(qū)類的allocateDirect(int capacity) 方法創(chuàng)建,此方法返回的緩沖區(qū)進行分配和取消分配所需的成本要高于非直接緩沖區(qū)。直接緩沖區(qū)的內(nèi)容駐留在垃圾回收堆之外,因此他們對應(yīng)用程序內(nèi)存(JVM內(nèi)存)需求不大。所以建議直接緩沖區(qū)要分配給那些大型,持久(就是緩沖區(qū)的數(shù)據(jù)會被重復(fù)利用)的緩沖區(qū),一般情況下,最好僅在直接緩沖區(qū)能在程序性能帶來非常明顯的好處時才分配它們。
- 直接緩沖區(qū)還可以通過FileCHannel的map()方法將文件區(qū)域映射到內(nèi)存中來創(chuàng)建,該方法返回MappedByteBuffer。java平臺的實現(xiàn)有助于通過JNI本地代碼創(chuàng)建直接字節(jié)緩沖區(qū),如果以上這些緩沖區(qū)中某個緩沖區(qū)實例指向的是不可訪問的內(nèi)存區(qū)域,則試圖方法該區(qū)域不會更改緩沖區(qū)的內(nèi)容,并且會在訪問期間或者稍后的某個時間導(dǎo)致報出不確定性異常。
- 字節(jié)緩沖區(qū)是直接緩沖區(qū)還是非直接緩沖區(qū)可以通過調(diào)用其isDIrect()方法來判斷。
基于NIO的本地IO直接內(nèi)存使用:
傳統(tǒng)IO對文件數(shù)據(jù)進行讀寫的流程:

流程說明(以上是應(yīng)用程序完成一次文件拷貝的流程):
- 應(yīng)用進程發(fā)起一個讀請求系統(tǒng)調(diào)用,然后進程切換到內(nèi)核態(tài)。
- DMA把磁盤數(shù)據(jù)復(fù)制到內(nèi)核緩沖區(qū)中。
- 內(nèi)核把緩沖區(qū)數(shù)據(jù)復(fù)制到用戶緩沖區(qū)中。
- 進程切換到用戶態(tài)。
- 應(yīng)用進程發(fā)起一個寫請求系統(tǒng)調(diào)用,然后進程切換到內(nèi)核態(tài)。
- 內(nèi)核把用戶緩沖區(qū)數(shù)據(jù)復(fù)制到內(nèi)核緩沖區(qū)。
- DMA把內(nèi)核緩沖區(qū)數(shù)據(jù)復(fù)制到磁盤上。
- 返回。
以上流程一共進行了四次上下文切換,四次數(shù)據(jù)拷貝。
使用mmap實現(xiàn)對傳統(tǒng)文件IO優(yōu)化。
mmap:通過把內(nèi)核緩沖區(qū)和用戶緩沖區(qū)映射在物理內(nèi)存上映射為同一地址空間。這樣就不用對數(shù)據(jù)進行復(fù)制了。

這個是傳統(tǒng)的:

使用mmap后的IO大致流程:

數(shù)據(jù)拷貝次數(shù)從4次縮短到了兩次。
相關(guān)API demo以及比較:詳細api解釋可以查看Java NIO學習篇之通道FileChannel詳解
//使用直接緩沖區(qū)API進行一個700多M的文件進行拷貝
public static void testDirect(){
try {
long start = System.currentTimeMillis();
FileChannel srcFileChannel = FileChannel.open(Paths.get("C:\\Users\\Yehaocong\\Desktop\\test\\95462017-1-64.flv"), StandardOpenOption.READ);
FileChannel destFileChannel = FileChannel.open(Paths.get("C:\\Users\\Yehaocong\\Desktop\\test\\95462017-1-64-cp1.flv"),StandardOpenOption.CREATE,
StandardOpenOption.WRITE,StandardOpenOption.READ);
MappedByteBuffer srcByteBuffer = srcFileChannel.map(FileChannel.MapMode.READ_ONLY,0,srcFileChannel.size());
MappedByteBuffer descByteBuffer = destFileChannel.map(FileChannel.MapMode.READ_WRITE,0,srcFileChannel.size());
descByteBuffer.put(srcByteBuffer);
srcFileChannel.close();
destFileChannel.close();
System.out.println("直接緩沖區(qū)耗時:" + (System.currentTimeMillis()-start));
} catch (IOException e) {
e.printStackTrace();
}
}
public static void testSimpleIO(){
try {
long start = System.currentTimeMillis();
FileChannel srcFileChannel = FileChannel.open(Paths.get("C:\\Users\\Yehaocong\\Desktop\\test\\95462017-1-64.flv"), StandardOpenOption.READ);
FileChannel destFileChannel = FileChannel.open(Paths.get("C:\\Users\\Yehaocong\\Desktop\\test\\95462017-1-64-cp.flv"),StandardOpenOption.CREATE,
StandardOpenOption.WRITE,StandardOpenOption.READ);
ByteBuffer byteBuffer = ByteBuffer.allocate((int) srcFileChannel.size());
while (srcFileChannel.read(byteBuffer)!=-1){
byteBuffer.flip();
destFileChannel.write(byteBuffer);
byteBuffer.clear();
}
srcFileChannel.close();
destFileChannel.close();
System.out.println("非緩沖區(qū)耗時:" + (System.currentTimeMillis()-start));
} catch (IOException e) {
e.printStackTrace();
}
}
執(zhí)行結(jié)果:
?

到此這篇關(guān)于淺析Java NIO 直接緩沖區(qū)和非直接緩沖區(qū)的文章就介紹到這了,更多相關(guān)Java NIO 直接緩沖區(qū)和非直接緩沖區(qū)內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
SpringSecurity+JWT實現(xiàn)前后端分離的使用詳解
這篇文章主要介紹了SpringSecurity+JWT實現(xiàn)前后端分離的使用詳解,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2021-01-01
springboot如何獲取相對路徑文件夾下靜態(tài)資源的方法
這篇文章主要介紹了springboot如何獲取相對路徑文件夾下靜態(tài)資源的方法,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2019-05-05
Java Spring Boot消息服務(wù)萬字詳解分析
在實際項目開發(fā)中,有時需要與其他系統(tǒng)進行集成完成相關(guān)業(yè)務(wù)功能,這種情況最原始做法是程序內(nèi)部相互調(diào)用,除此之外,還可以用消息服務(wù)中間件進行業(yè)務(wù)處理,用消息服務(wù)中間件處理業(yè)務(wù)能夠提升系統(tǒng)的異步通信和擴展解耦能力。Spring Boot對消息服務(wù)管理提供了非常好的支持2021-10-10
解決Java字符串JSON轉(zhuǎn)換異常:cn.hutool.json.JSONException:?Mismatched?
這篇文章主要給大家介紹了關(guān)于如何解決Java字符串JSON轉(zhuǎn)換異常:cn.hutool.json.JSONException:?Mismatched?hr?and?body的相關(guān)資料,文中將解決的辦法通過代碼介紹的非常詳細,需要的朋友可以參考下2024-01-01
SpringSecurity+jwt+redis基于數(shù)據(jù)庫登錄認證的實現(xiàn)
本文主要介紹了SpringSecurity+jwt+redis基于數(shù)據(jù)庫登錄認證的實現(xiàn),其中也涉及到自定義的過濾器和處理器,具有一定的參考價值,感興趣的可以了解一下2023-09-09
Spring?Boot在啟動時執(zhí)行一次的功能實現(xiàn)
這篇文章主要給大家介紹了關(guān)于Spring?Boot在啟動時執(zhí)行一次的功能實現(xiàn),在實習過程中,有時候會遇到一些項目啟動初始化的需求,文中通過示例代碼介紹的非常詳細,需要的朋友可以參考下2023-08-08
SpringBoot+LayIM+t-io 實現(xiàn)好友申請通知流程
這篇文章主要介紹了 SpringBoot+LayIM+t-io 實現(xiàn)好友申請通知流程,本文圖文并茂給大家介紹的非常詳細,需要的朋友可以參考下2017-12-12

