欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java NIO工作原理的全面分析

 更新時(shí)間:2013年02月19日 11:28:55   作者:  
JDK 1.4 中引入的新輸入輸出 (NIO) 庫在標(biāo)準(zhǔn) Java 代碼中提供了高速的、面向塊的 I/O。本實(shí)用教程從高級概念到底層的編程細(xì)節(jié),非常詳細(xì)地介紹了 NIO 庫。您將學(xué)到諸如緩沖區(qū)和通道這樣的關(guān)鍵 I/O 元素的知識,并考察更新后的庫中的標(biāo)準(zhǔn) I/O 是如何工作的。您還將了解只能通過 NIO 來完成的工作,如異步 I/O 和直接緩沖區(qū)。
◆  輸入/輸出:概念性描述
I/O 簡介
I/O ? 或者輸入/輸出 ? 指的是計(jì)算機(jī)與外部世界或者一個(gè)程序與計(jì)算機(jī)的其余部分的之間的接口。它對于任何計(jì)算機(jī)系統(tǒng)都非常關(guān)鍵,因而所有 I/O 的主體實(shí)際上是內(nèi)置在操作系統(tǒng)中的。單獨(dú)的程序一般是讓系統(tǒng)為它們完成大部分的工作。
在 Java 編程中,直到最近一直使用 流 的方式完成 I/O。所有 I/O 都被視為單個(gè)的字節(jié)的移動(dòng),通過一個(gè)稱為 Stream 的對象一次移動(dòng)一個(gè)字節(jié)。流 I/O 用于與外部世界接觸。它也在內(nèi)部使用,用于將對象轉(zhuǎn)換為字節(jié),然后再轉(zhuǎn)換回對象。
NIO 與原來的 I/O 有同樣的作用和目的,但是它使用不同的方式? 塊 I/O。正如您將在本教程中學(xué)到的,塊 I/O 的效率可以比流 I/O 高許多。
為什么要使用 NIO?
NIO 的創(chuàng)建目的是為了讓 Java 程序員可以實(shí)現(xiàn)高速 I/O 而無需編寫自定義的本機(jī)代碼。NIO 將最耗時(shí)的 I/O 操作(即填充和提取緩沖區(qū))轉(zhuǎn)移回操作系統(tǒng),因而可以極大地提高速度。
流與塊的比較
原來的 I/O 庫(在 java.io.*中) 與 NIO 最重要的區(qū)別是數(shù)據(jù)打包和傳輸?shù)姆绞?。正如前面提到的,原來?I/O 以流的方式處理數(shù)據(jù),而 NIO 以塊的方式處理數(shù)據(jù)。
面向流 的 I/O 系統(tǒng)一次一個(gè)字節(jié)地處理數(shù)據(jù)。一個(gè)輸入流產(chǎn)生一個(gè)字節(jié)的數(shù)據(jù),一個(gè)輸出流消費(fèi)一個(gè)字節(jié)的數(shù)據(jù)。為流式數(shù)據(jù)創(chuàng)建過濾器非常容易。鏈接幾個(gè)過濾器,以便每個(gè)過濾器只負(fù)責(zé)單個(gè)復(fù)雜處理機(jī)制的一部分,這樣也是相對簡單的。不利的一面是,面向流的 I/O 通常相當(dāng)慢。
一個(gè) 面向塊 的 I/O 系統(tǒng)以塊的形式處理數(shù)據(jù)。每一個(gè)操作都在一步中產(chǎn)生或者消費(fèi)一個(gè)數(shù)據(jù)塊。按塊處理數(shù)據(jù)比按(流式的)字節(jié)處理數(shù)據(jù)要快得多。但是面向塊的 I/O 缺少一些面向流的 I/O 所具有的優(yōu)雅性和簡單性。
集成的 I/O
在 JDK 1.4 中原來的 I/O 包和 NIO 已經(jīng)很好地集成了。 java.io.* 已經(jīng)以 NIO 為基礎(chǔ)重新實(shí)現(xiàn)了,所以現(xiàn)在它可以利用 NIO 的一些特性。例如, java.io.* 包中的一些類包含以塊的形式讀寫數(shù)據(jù)的方法,這使得即使在更面向流的系統(tǒng)中,處理速度也會更快。
也可以用 NIO 庫實(shí)現(xiàn)標(biāo)準(zhǔn) I/O 功能。例如,可以容易地使用塊 I/O 一次一個(gè)字節(jié)地移動(dòng)數(shù)據(jù)。但是正如您會看到的,NIO 還提供了原 I/O 包中所沒有的許多好處。
◆ 通道和緩沖區(qū)
概  述
通道 和 緩沖區(qū) 是 NIO 中的核心對象,幾乎在每一個(gè) I/O 操作中都要使用它們。
通道是對原 I/O 包中的流的模擬。到任何目的地(或來自任何地方)的所有數(shù)據(jù)都必須通過一個(gè) Channel 對象。一個(gè) Buffer 實(shí)質(zhì)上是一個(gè)容器對象。發(fā)送給一個(gè)通道的所有對象都必須首先放到緩沖區(qū)中;同樣地,從通道中讀取的任何數(shù)據(jù)都要讀到緩沖區(qū)中。
在本節(jié)中,您會了解到 NIO 中通道和緩沖區(qū)是如何工作的。
什么是緩沖區(qū)?
Buffer 是一個(gè)對象, 它包含一些要寫入或者剛讀出的數(shù)據(jù)。 在 NIO 中加入 Buffer 對象,體現(xiàn)了新庫與原 I/O 的一個(gè)重要區(qū)別。在面向流的 I/O 中,您將數(shù)據(jù)直接寫入或者將數(shù)據(jù)直接讀到 Stream 對象中。
在 NIO 庫中,所有數(shù)據(jù)都是用緩沖區(qū)處理的。在讀取數(shù)據(jù)時(shí),它是直接讀到緩沖區(qū)中的。在寫入數(shù)據(jù)時(shí),它是寫入到緩沖區(qū)中的。任何時(shí)候訪問 NIO 中的數(shù)據(jù),您都是將它放到緩沖區(qū)中。
緩沖區(qū)實(shí)質(zhì)上是一個(gè)數(shù)組。通常它是一個(gè)字節(jié)數(shù)組,但是也可以使用其他種類的數(shù)組。但是一個(gè)緩沖區(qū)不 僅僅 是一個(gè)數(shù)組。緩沖區(qū)提供了對數(shù)據(jù)的結(jié)構(gòu)化訪問,而且還可以跟蹤系統(tǒng)的讀/寫進(jìn)程。
緩沖區(qū)類型
最常用的緩沖區(qū)類型是 ByteBuffer。一個(gè) ByteBuffer 可以在其底層字節(jié)數(shù)組上進(jìn)行 get/set 操作(即字節(jié)的獲取和設(shè)置)。
ByteBuffer 不是 NIO 中唯一的緩沖區(qū)類型。事實(shí)上,對于每一種基本 Java 類型都有一種緩沖區(qū)類型:
• ByteBuffer
• CharBuffer
• ShortBuffer
• IntBuffer
• LongBuffer
• FloatBuffer
• DoubleBuffer
每一個(gè) Buffer 類都是 Buffer 接口的一個(gè)實(shí)例。 除了 ByteBuffer,每一個(gè) Buffer 類都有完全一樣的操作,只是它們所處理的數(shù)據(jù)類型不一樣。因?yàn)榇蠖鄶?shù)標(biāo)準(zhǔn) I/O 操作都使用 ByteBuffer,所以它具有所有共享的緩沖區(qū)操作以及一些特有的操作。
現(xiàn)在您可以花一點(diǎn)時(shí)間運(yùn)行 UseFloatBuffer.java,它包含了類型化的緩沖區(qū)的一個(gè)應(yīng)用例子。
什么是通道?
Channel是一個(gè)對象,可以通過它讀取和寫入數(shù)據(jù)。拿 NIO 與原來的 I/O 做個(gè)比較,通道就像是流。
正如前面提到的,所有數(shù)據(jù)都通過 Buffer 對象來處理。您永遠(yuǎn)不會將字節(jié)直接寫入通道中,相反,您是將數(shù)據(jù)寫入包含一個(gè)或者多個(gè)字節(jié)的緩沖區(qū)。同樣,您不會直接從通道中讀取字節(jié),而是將數(shù)據(jù)從通道讀入緩沖區(qū),再從緩沖區(qū)獲取這個(gè)字節(jié)。
通道類型
通道與流的不同之處在于通道是雙向的。而流只是在一個(gè)方向上移動(dòng)(一個(gè)流必須是 InputStream 或者 OutputStream 的子類), 而 通道 可以用于讀、寫或者同時(shí)用于讀寫。
因?yàn)樗鼈兪请p向的,所以通道可以比流更好地反映底層操作系統(tǒng)的真實(shí)情況。特別是在 UNIX 模型中,底層操作系統(tǒng)通道是雙向的。
◆ 從理論到實(shí)踐:NIO 中的讀和寫
概  述
讀和寫是 I/O 的基本過程。從一個(gè)通道中讀取很簡單:只需創(chuàng)建一個(gè)緩沖區(qū),然后讓通道將數(shù)據(jù)讀到這個(gè)緩沖區(qū)中。寫入也相當(dāng)簡單:創(chuàng)建一個(gè)緩沖區(qū),用數(shù)據(jù)填充它,然后讓通道用這些數(shù)據(jù)來執(zhí)行寫入操作。
在本節(jié)中,我們將學(xué)習(xí)有關(guān)在 Java 程序中讀取和寫入數(shù)據(jù)的一些知識。我們將回顧 NIO 的主要組件(緩沖區(qū)、通道和一些相關(guān)的方法),看看它們是如何交互以進(jìn)行讀寫的。在接下來的幾節(jié)中,我們將更詳細(xì)地分析這其中的每個(gè)組件以及其交互。
從文件中讀取
在我們第一個(gè)練習(xí)中,我們將從一個(gè)文件中讀取一些數(shù)據(jù)。如果使用原來的 I/O,那么我們只需創(chuàng)建一個(gè) FileInputStream 并從它那里讀取。而在 NIO 中,情況稍有不同:我們首先從 FileInputStream 獲取一個(gè) FileInputStream 對象,然后使用這個(gè)通道來讀取數(shù)據(jù)。
在 NIO 系統(tǒng)中,任何時(shí)候執(zhí)行一個(gè)讀操作,您都是從通道中讀取,但是您不是 直接 從通道讀取。因?yàn)樗袛?shù)據(jù)最終都駐留在緩沖區(qū)中,所以您是從通道讀到緩沖區(qū)中。
因此讀取文件涉及三個(gè)步驟:(1) 從 FileInputStream 獲取 Channel,(2) 創(chuàng)建 Buffer,(3) 將數(shù)據(jù)從 Channel 讀到 Buffer 中。
現(xiàn)在,讓我們看一下這個(gè)過程。
三個(gè)容易的步驟
第一步是獲取通道。我們從 FileInputStream 獲取通道:
復(fù)制代碼 代碼如下:

FileInputStream fin = new FileInputStream( "readandshow.txt" ); 
FileChannel fc = fin.getChannel();

下一步是創(chuàng)建緩沖區(qū):
復(fù)制代碼 代碼如下:

ByteBuffer buffer = ByteBuffer.allocate( 1024 );

最后,需要將數(shù)據(jù)從通道讀到緩沖區(qū)中,如下所示:
復(fù)制代碼 代碼如下:

fc.read( buffer );

您會注意到,我們不需要告訴通道要讀 多少數(shù)據(jù) 到緩沖區(qū)中。每一個(gè)緩沖區(qū)都有復(fù)雜的內(nèi)部統(tǒng)計(jì)機(jī)制,它會跟蹤已經(jīng)讀了多少數(shù)據(jù)以及還有多少空間可以容納更多的數(shù)據(jù)
寫入文件
在 NIO 中寫入文件類似于從文件中讀取。首先從 FileOutputStream 獲取一個(gè)通道:
復(fù)制代碼 代碼如下:

FileOutputStream fout = new FileOutputStream( "writesomebytes.txt" ); 
FileChannel fc = fout.getChannel();

下一步是創(chuàng)建一個(gè)緩沖區(qū)并在其中放入一些數(shù)據(jù) - 在這里,數(shù)據(jù)將從一個(gè)名為 message 的數(shù)組中取出,這個(gè)數(shù)組包含字符串 "Some bytes" 的 ASCII 字節(jié)(本教程后面將會解釋 buffer.flip() 和 buffer.put() 調(diào)用)。
復(fù)制代碼 代碼如下:

ByteBuffer buffer = ByteBuffer.allocate( 1024 ); 
 for (int i=0; i
     buffer.put( message[i] ); 

buffer.flip();

最后一步是寫入緩沖區(qū)中
復(fù)制代碼 代碼如下:

fc.write( buffer );

注意在這里同樣不需要告訴通道要寫入多數(shù)據(jù)。緩沖區(qū)的內(nèi)部統(tǒng)計(jì)機(jī)制會跟蹤它包含多少數(shù)據(jù)以及還有多少數(shù)據(jù)要寫入。
讀寫結(jié)合
下面我們將看一下在結(jié)合讀和寫時(shí)會有什么情況。我們以一個(gè)名為 CopyFile.java 的簡單程序作為這個(gè)練習(xí)的基礎(chǔ),它將一個(gè)文件的所有內(nèi)容拷貝到另一個(gè)文件中。CopyFile.java 執(zhí)行三個(gè)基本操作:首先創(chuàng)建一個(gè) Buffer,然后從源文件中將數(shù)據(jù)讀到這個(gè)緩沖區(qū)中,然后將緩沖區(qū)寫入目標(biāo)文件。這個(gè)程序不斷重復(fù) ― 讀、寫、讀、寫 ― 直到源文件結(jié)束。
CopyFile 程序讓您看到我們?nèi)绾螜z查操作的狀態(tài),以及如何使用 clear() 和 flip() 方法重設(shè)緩沖區(qū),并準(zhǔn)備緩沖區(qū)以便將新讀取的數(shù)據(jù)寫到另一個(gè)通道中。
運(yùn)行 CopyFile 例子
因?yàn)榫彌_區(qū)會跟蹤它自己的數(shù)據(jù),所以 CopyFile 程序的內(nèi)部循環(huán) (inner loop) 非常簡單,如下所示:
復(fù)制代碼 代碼如下:

fcin.read( buffer ); 
fcout.write( buffer );

第一行將數(shù)據(jù)從輸入通道 fcin 中讀入緩沖區(qū),第二行將這些數(shù)據(jù)寫到輸出通道 fcout 。
檢查狀態(tài)
下一步是檢查拷貝何時(shí)完成。當(dāng)沒有更多的數(shù)據(jù)時(shí),拷貝就算完成,并且可以在 read() 方法返回 -1 是判斷這一點(diǎn),如下所示:
復(fù)制代碼 代碼如下:

int r = fcin.read( buffer ); 
 if (r==-1) { 
     break; 
}

重設(shè)緩沖區(qū)
最后,在從輸入通道讀入緩沖區(qū)之前,我們調(diào)用 clear() 方法。同樣,在將緩沖區(qū)寫入輸出通道之前,我們調(diào)用 flip() 方法,如下所示
復(fù)制代碼 代碼如下:

buffer.clear();int r = fcin.read( buffer ); 
 if (r==-1) { 
     break; 

 buffer.flip(); 
fcout.write( buffer );

clear() 方法重設(shè)緩沖區(qū),使它可以接受讀入的數(shù)據(jù)。 flip() 方法讓緩沖區(qū)可以將新讀入的數(shù)據(jù)寫入另一個(gè)通道。

相關(guān)文章

  • rocketmq client 日志的問題處理方式

    rocketmq client 日志的問題處理方式

    這篇文章主要介紹了rocketmq client 日志的問題處理方式,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-10-10
  • maven搭建spring項(xiàng)目(圖文教程)

    maven搭建spring項(xiàng)目(圖文教程)

    下面小編就為大家?guī)硪黄猰aven搭建spring項(xiàng)目(圖文教程)。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2017-06-06
  • SpringMVC 參數(shù)綁定相關(guān)知識總結(jié)

    SpringMVC 參數(shù)綁定相關(guān)知識總結(jié)

    這篇文章主要介紹了SpringMVC 參數(shù)綁定相關(guān)知識總結(jié),幫助大家更好的理解和學(xué)習(xí)使用SpringMVC,感興趣的朋友可以了解下
    2021-03-03
  • 一篇文章詳解Java異常處理

    一篇文章詳解Java異常處理

    異常處理是編程語言或計(jì)算機(jī)硬件里的一種機(jī)制,用于處理軟件或信息系統(tǒng)中出現(xiàn)的異常狀況(即超出程序正常執(zhí)行流程的某些特殊條件),這篇文章主要給大家介紹了關(guān)于Java異常處理的相關(guān)資料,需要的朋友可以參考下
    2023-12-12
  • Java反射機(jī)制詳解_動(dòng)力節(jié)點(diǎn)Java學(xué)院整理

    Java反射機(jī)制詳解_動(dòng)力節(jié)點(diǎn)Java學(xué)院整理

    這篇文章主要為大家詳細(xì)介紹了Java反射機(jī)制的相關(guān)資料,主要包括反射的概念、作用
    2017-06-06
  • Springboot 1.5.7整合Kafka-client代碼示例

    Springboot 1.5.7整合Kafka-client代碼示例

    這篇文章主要介紹了Springboot 1.5.7整合Kafka-client代碼示例,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-10-10
  • 一篇文章帶你了解Java容器,面板及四大布局管理器應(yīng)用

    一篇文章帶你了解Java容器,面板及四大布局管理器應(yīng)用

    這篇文章主要介紹了JAVA布局管理器與面板組合代碼實(shí)例,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2021-08-08
  • SpringBoot中web模版數(shù)據(jù)渲染展示的案例詳解

    SpringBoot中web模版數(shù)據(jù)渲染展示的案例詳解

    憑借 Spring Framework 的模塊、與你最喜歡的工具的大量集成以及插入你自己的功能的能力,Thymeleaf 是現(xiàn)代 HTML5 JVM Web 開發(fā)的理想選擇——盡管它還有更多功能,本文重點(diǎn)給大家介紹SpringBoot中web模版數(shù)據(jù)渲染展示,需要的朋友可以參考下
    2022-01-01
  • Springboot 跨域配置無效及接口訪問報(bào)錯(cuò)的解決方法

    Springboot 跨域配置無效及接口訪問報(bào)錯(cuò)的解決方法

    這篇文章主要介紹了Springboot 跨域配置無效及接口訪問報(bào)錯(cuò)的解決方法,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-12-12
  • SpringMVC打印請求參數(shù)和響應(yīng)數(shù)據(jù)最優(yōu)方案

    SpringMVC打印請求參數(shù)和響應(yīng)數(shù)據(jù)最優(yōu)方案

    項(xiàng)目中經(jīng)常需要打印http請求的參數(shù)和響應(yīng)數(shù)據(jù),本文給大家講解如何在SpringMVC打印請求參數(shù)和響應(yīng)數(shù)據(jù)最優(yōu)方案,感興趣的朋友跟隨小編一起看看吧
    2023-07-07

最新評論