小編給大家分享一下Java NIO API有哪些,相信大部分人都還不怎么了解,因此分享這篇文章給大家參考一下,希望大家閱讀完這篇文章后大有收獲,下面讓我們一起去了解一下吧!
望城網(wǎng)站制作公司哪家好,找創(chuàng)新互聯(lián)公司!從網(wǎng)頁(yè)設(shè)計(jì)、網(wǎng)站建設(shè)、微信開(kāi)發(fā)、APP開(kāi)發(fā)、成都響應(yīng)式網(wǎng)站建設(shè)公司等網(wǎng)站項(xiàng)目制作,到程序開(kāi)發(fā),運(yùn)營(yíng)維護(hù)。創(chuàng)新互聯(lián)公司2013年至今到現(xiàn)在10年的時(shí)間,我們擁有了豐富的建站經(jīng)驗(yàn)和運(yùn)維經(jīng)驗(yàn),來(lái)保證我們的工作的順利進(jìn)行。專注于網(wǎng)站建設(shè)就選創(chuàng)新互聯(lián)公司。
NIO API主要集中在java.nio和它的subpackages中:
java.nio
定義了Buffer及其數(shù)據(jù)類型相關(guān)的子類。其中被java.nio.channels中的類用來(lái)進(jìn)行IO操作的ByteBuffer的作用非常重要。
java.nio.channels
定義了一系列處理IO的Channel接口以及這些接口在文件系統(tǒng)和網(wǎng)絡(luò)通訊上的實(shí)現(xiàn)。通過(guò)Selector這個(gè)類,還提供了進(jìn)行異步IO操作的辦法。這個(gè)包可以說(shuō)是NIO API的核心。
java.nio.channels.spi
定義了可用來(lái)實(shí)現(xiàn)channel和selector API的抽象類。
java.nio.charset
定義了處理字符編碼和解碼的類。
java.nio.charset.spi
定義了可用來(lái)實(shí)現(xiàn)charset API的抽象類。
java.nio.channels.spi和java.nio.charset.spi這兩個(gè)包主要被用來(lái)對(duì)現(xiàn)有NIO API進(jìn)行擴(kuò)展,在實(shí)際的使用中,我們一般只和另外的3個(gè)包打交道。下面將對(duì)這3個(gè)包一一介紹。
Package java.nio
這個(gè)包主要定義了Buffer及其子類。Buffer定義了一個(gè)線性存放primitive type數(shù)據(jù)的容器接口。對(duì)于除boolean以外的其他primitive type,都有一個(gè)相應(yīng)的Buffer子類,ByteBuffer是其中最重要的一個(gè)子類。
下面這張UML類圖描述了java.nio中的類的關(guān)系:
Buffer
定義了一個(gè)可以線性存放primitive type數(shù)據(jù)的容器接口。Buffer主要包含了與類型(byte, char…)無(wú)關(guān)的功能。值得注意的是Buffer及其子類都不是線程安全的。
每個(gè)Buffer都有以下的屬性:
capacity
這個(gè)Buffer最多能放多少數(shù)據(jù)。capacity一般在buffer被創(chuàng)建的時(shí)候指定。
limit
在Buffer上進(jìn)行的讀寫(xiě)操作都不能越過(guò)這個(gè)下標(biāo)。當(dāng)寫(xiě)數(shù)據(jù)到buffer中時(shí),limit一般和capacity相等,當(dāng)讀數(shù)據(jù)時(shí),limit代表buffer中有效數(shù)據(jù)的長(zhǎng)度。
position
讀/寫(xiě)操作的當(dāng)前下標(biāo)。當(dāng)使用buffer的相對(duì)位置進(jìn)行讀/寫(xiě)操作時(shí),讀/寫(xiě)會(huì)從這個(gè)下標(biāo)進(jìn)行,并在操作完成后,buffer會(huì)更新下標(biāo)的值。
mark
一個(gè)臨時(shí)存放的位置下標(biāo)。調(diào)用mark()會(huì)將mark設(shè)為當(dāng)前的position的值,以后調(diào)用reset()會(huì)將position屬性設(shè)置為mark的值。mark的值總是小于等于position的值,如果將position的值設(shè)的比mark小,當(dāng)前的mark值會(huì)被拋棄掉。
這些屬性總是滿足以下條件:
0 <= mark <= position <= limit <= capacity
limit和position的值除了通過(guò)limit()和position()函數(shù)來(lái)設(shè)置,也可以通過(guò)下面這些函數(shù)來(lái)改變:
Buffer clear()
把position設(shè)為0,把limit設(shè)為capacity,一般在把數(shù)據(jù)寫(xiě)入Buffer前調(diào)用。
Buffer flip()
把limit設(shè)為當(dāng)前position,把position設(shè)為0,一般在從Buffer讀出數(shù)據(jù)前調(diào)用。
Buffer rewind()
把position設(shè)為0,limit不變,一般在把數(shù)據(jù)重寫(xiě)入Buffer前調(diào)用。
Buffer對(duì)象有可能是只讀的,這時(shí),任何對(duì)該對(duì)象的寫(xiě)操作都會(huì)觸發(fā)一個(gè)ReadOnlyBufferException。isReadOnly()方法可以用來(lái)判斷一個(gè)Buffer是否只讀。
ByteBuffer
在Buffer的子類中,ByteBuffer是一個(gè)地位較為特殊的類,因?yàn)樵趈ava.io.channels中定義的各種channel的IO操作基本上都是圍繞ByteBuffer展開(kāi)的。
ByteBuffer定義了4個(gè)static方法來(lái)做創(chuàng)建工作:
ByteBuffer allocate(int capacity)
創(chuàng)建一個(gè)指定capacity的ByteBuffer。
ByteBuffer allocateDirect(int capacity)
創(chuàng)建一個(gè)direct的ByteBuffer,這樣的ByteBuffer在參與IO操作時(shí)性能會(huì)更好(很有可能是在底層的實(shí)現(xiàn)使用了DMA技術(shù)),相應(yīng)的,創(chuàng)建和回收direct的ByteBuffer的代價(jià)也會(huì)高一些。isDirect()方法可以檢查一個(gè)buffer是否是direct的。
ByteBuffer wrap(byte [] array)
ByteBuffer wrap(byte [] array, int offset, int length)
把一個(gè)byte數(shù)組或byte數(shù)組的一部分包裝成ByteBuffer。
ByteBuffer定義了一系列g(shù)et和put操作來(lái)從中讀寫(xiě)byte數(shù)據(jù),如下面幾個(gè):
byte get() ByteBuffer get(byte [] dst) byte get(int index) ByteBuffer put(byte b) ByteBuffer put(byte [] src) ByteBuffer put(int index, byte b)
這些操作可分為絕對(duì)定位和相對(duì)定為兩種,相對(duì)定位的讀寫(xiě)操作依靠position來(lái)定位Buffer中的位置,并在操作完成后會(huì)更新position的值。
在其它類型的buffer中,也定義了相同的函數(shù)來(lái)讀寫(xiě)數(shù)據(jù),***不同的就是一些參數(shù)和返回值的類型。
除了讀寫(xiě)byte類型數(shù)據(jù)的函數(shù),ByteBuffer的一個(gè)特別之處是它還定義了讀寫(xiě)其它primitive數(shù)據(jù)的方法,如:
int getInt()
從ByteBuffer中讀出一個(gè)int值。
ByteBuffer putInt(int value)
寫(xiě)入一個(gè)int值到ByteBuffer中。
讀寫(xiě)其它類型的數(shù)據(jù)牽涉到字節(jié)序問(wèn)題,ByteBuffer會(huì)按其字節(jié)序(大字節(jié)序或小字節(jié)序)寫(xiě)入或讀出一個(gè)其它類型的數(shù)據(jù)(int,long…)。字節(jié)序可以用order方法來(lái)取得和設(shè)置:
ByteOrder order()
返回ByteBuffer的字節(jié)序。
ByteBuffer order(ByteOrder bo)
設(shè)置ByteBuffer的字節(jié)序。
ByteBuffer另一個(gè)特別的地方是可以在它的基礎(chǔ)上得到其它類型的buffer。如:
CharBuffer asCharBuffer()
為當(dāng)前的ByteBuffer創(chuàng)建一個(gè)CharBuffer的視圖。在該視圖buffer中的讀寫(xiě)操作會(huì)按照ByteBuffer的字節(jié)序作用到ByteBuffer中的數(shù)據(jù)上。
用這類方法創(chuàng)建出來(lái)的buffer會(huì)從ByteBuffer的position位置開(kāi)始到limit位置結(jié)束,可以看作是這段數(shù)據(jù)的視圖。視圖buffer的readOnly屬性和direct屬性與ByteBuffer的一致,而且也只有通過(guò)這種方法,才可以得到其他數(shù)據(jù)類型的direct buffer。
ByteOrder
用來(lái)表示ByteBuffer字節(jié)序的類,可將其看成java中的enum類型。主要定義了下面幾個(gè)static方法和屬性:
ByteOrder BIG_ENDIAN
代表大字節(jié)序的ByteOrder。
ByteOrder LITTLE_ENDIAN
代表小字節(jié)序的ByteOrder。
ByteOrder nativeOrder()
返回當(dāng)前硬件平臺(tái)的字節(jié)序。
MappedByteBuffer
ByteBuffer的子類,是文件內(nèi)容在內(nèi)存中的映射。這個(gè)類的實(shí)例需要通過(guò)FileChannel的map()方法來(lái)創(chuàng)建。
接下來(lái)看看一個(gè)使用ByteBuffer的例子,這個(gè)例子從標(biāo)準(zhǔn)輸入不停地讀入字符,當(dāng)讀滿一行后,將收集的字符寫(xiě)到標(biāo)準(zhǔn)輸出:
public static void main(String [] args) throws IOException { // 創(chuàng)建一個(gè)capacity為256的ByteBuffer ByteBuffer buf = ByteBuffer.allocate(256); while (true) { // 從標(biāo)準(zhǔn)輸入流讀入一個(gè)字符 int c = System.in.read(); // 當(dāng)讀到輸入流結(jié)束時(shí),退出循環(huán) if (c == -1) break; // 把讀入的字符寫(xiě)入ByteBuffer中 buf.put((byte) c); // 當(dāng)讀完一行時(shí),輸出收集的字符 if (c == '\n') { // 調(diào)用flip()使limit變?yōu)楫?dāng)前的position的值,position變?yōu)?, // 為接下來(lái)從ByteBuffer讀取做準(zhǔn)備 buf.flip(); // 構(gòu)建一個(gè)byte數(shù)組 byte [] content = new byte[buf.limit()]; // 從ByteBuffer中讀取數(shù)據(jù)到byte數(shù)組中 buf.get(content); // 把byte數(shù)組的內(nèi)容寫(xiě)到標(biāo)準(zhǔn)輸出 System.out.print(new String(content)); // 調(diào)用clear()使position變?yōu)?,limit變?yōu)閏apacity的值, // 為接下來(lái)寫(xiě)入數(shù)據(jù)到ByteBuffer中做準(zhǔn)備 buf.clear(); } } }
Package java.nio.channels
這個(gè)包定義了Channel的概念,Channel表現(xiàn)了一個(gè)可以進(jìn)行IO操作的通道(比如,通過(guò)FileChannel,我們可以對(duì)文件進(jìn)行讀寫(xiě)操作)。java.nio.channels包含了文件系統(tǒng)和網(wǎng)絡(luò)通訊相關(guān)的channel類。這個(gè)包通過(guò)Selector和SelectableChannel這兩個(gè)類,還定義了一個(gè)進(jìn)行異步(non-blocking)IO操作的API,這對(duì)需要高性能IO的應(yīng)用非常重要。
下面這張UML類圖描述了java.nio.channels中interface的關(guān)系:
Channel
Channel表現(xiàn)了一個(gè)可以進(jìn)行IO操作的通道,該interface定義了以下方法:
boolean isOpen()
該Channel是否是打開(kāi)的。
void close()
關(guān)閉這個(gè)Channel,相關(guān)的資源會(huì)被釋放。
ReadableByteChannel
定義了一個(gè)可從中讀取byte數(shù)據(jù)的channel interface。
int read(ByteBuffer dst)
從channel中讀取byte數(shù)據(jù)并寫(xiě)到ByteBuffer中。返回讀取的byte數(shù)。
WritableByteChannel
定義了一個(gè)可向其寫(xiě)byte數(shù)據(jù)的channel interface。
int write(ByteBuffer src)
從ByteBuffer中讀取byte數(shù)據(jù)并寫(xiě)到channel中。返回寫(xiě)出的byte數(shù)。
ByteChannel
ByteChannel并沒(méi)有定義新的方法,它的作用只是把ReadableByteChannel和WritableByteChannel合并在一起。
ScatteringByteChannel
繼承了ReadableByteChannel并提供了同時(shí)往幾個(gè)ByteBuffer中寫(xiě)數(shù)據(jù)的能力。
GatheringByteChannel
繼承了WritableByteChannel并提供了同時(shí)從幾個(gè)。
以上是“Java NIO API有哪些”這篇文章的所有內(nèi)容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內(nèi)容對(duì)大家有所幫助,如果還想學(xué)習(xí)更多知識(shí),歡迎關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道!