Java NIO

參考鏈接:
Java NIO Tutorial

1. Java NIO 概覽:

NIO:Non-block IO,非阻塞IO。
Java NIO主要包括三個核心組件:

  1. Channels
  2. Buffers
  3. Selectors

Channels和Buffers
NIO中所有的IO都從一個Channel開始,Channel就像一個數(shù)據(jù)流誊册,我們可以從Channel中讀取數(shù)據(jù)到Buffer中仙蛉,也可以從Buffer通過Channel寫出數(shù)據(jù)材彪。如圖所示:

Paste_Image.png

在NIO中有多種類型的Channel和Buffer您市,一些基本的Channel實現(xiàn)如下:

  • FileChannel
  • DatagramChannel
  • SocketChannel
  • ServerSocketChannel

可以看出循榆,這些Channel包含了TCP和UDP協(xié)議的網(wǎng)絡(luò)IO,以及文件IO墨坚。
接下來列出一些NIO中的核心Buffer:

  • ByteBuffer
  • CharBuffer
  • DoubleBuffer
  • FloatBuffer
  • IntBuffer
  • LongBuffer
  • ShortBuffer

你可以通過這些Buffer傳送byte秧饮,character,等基本類型的數(shù)據(jù)泽篮。

Selectors
NIO中通過Selector負責(zé)監(jiān)聽多個Channel的事件盗尸,例如連接建立,數(shù)據(jù)到達等帽撑,這樣一來泼各,我們就可以在單線程中非阻塞地處理多個Channel中的數(shù)據(jù)。
我們會在每個Selector中注冊一個或多個Channel亏拉,調(diào)用selector的select()方法進行事件處理扣蜻。

2. Channel

四種Channel:

  • FileChannel : 從文件讀寫數(shù)據(jù)

  • DatagramChannel : 讀寫通過UDP協(xié)議傳輸?shù)木W(wǎng)絡(luò)數(shù)據(jù)

  • SocketChannel:處理通過TCP協(xié)議傳輸?shù)木W(wǎng)絡(luò)數(shù)據(jù)

  • ServerSocketChannel:使得我們能夠監(jiān)聽到達服務(wù)器的TCP連接請求,就像Web服務(wù)器一樣及塘。每有一個鏈接到來就會創(chuàng)建并分配一個 SocketChannel對象莽使。
    下面的代碼展示了一個使用FileChannel將一些數(shù)據(jù)寫入到一個ByteBuffer中:
    Java
    RandomAccessFile aFile = new RandomAccessFile("data/nio-data.txt", "rw");
    FileChannel inChannel = aFile.getChannel();

    ByteBuffer buf = ByteBuffer.allocate(48);

    int bytesRead = inChannel.read(buf);
    while (bytesRead != -1) {

    System.out.println("Read " + bytesRead);
    buf.flip();
    
    while(buf.hasRemaining()){
        System.out.print((char) buf.get());
    }
    
    buf.clear();
    bytesRead = inChannel.read(buf);
    

    }
    aFile.close();


## 3. Buffer
buffer 其實就是一塊你可以進行讀寫數(shù)據(jù)的內(nèi)存。這塊內(nèi)存被NIO Buffer對象封裝笙僚,該對象提供一系列API使得處理內(nèi)存更加容易芳肌。
使用Buffer進行數(shù)據(jù)讀寫一般要遵循以下4步:
- 寫數(shù)據(jù)到buffer中
- 調(diào)用buffer.flip()將buffer從寫入模式改為讀取模式
- 讀數(shù)據(jù)
- 調(diào)用buffer.clear()清空數(shù)據(jù)或者buffer.compact()清除你所讀取的那部分數(shù)據(jù)

下面是一個栗子:
```Java```
RandomAccessFile aFile = new RandomAccessFile("data/nio-data.txt", "rw");
FileChannel inChannel = aFile.getChannel();
//create buffer with capacity of 48 bytes
ByteBuffer buf = ByteBuffer.allocate(48);
int bytesRead = inChannel.read(buf); //read into buffer.
while (bytesRead != -1) { 
    buf.flip(); //make buffer ready for read 
    while(buf.hasRemaining()){ 
      System.out.print((char) buf.get()); // read 1 byte at a time 
    } 
    buf.clear();*//make buffer ready for writing 
    bytesRead = inChannel.read(buf);
}
aFile.close();

4.Selector

如下代碼可以創(chuàng)建一個Selector:
Selector selector = Selector.open()

將Selector注冊到Channel:

channel.configureBlocking(false);
SelectionKey key = channel.register(selector, SelectionKey.OP_READ);

我們注意到注冊方法第二個參數(shù)代表著一個讀取操作,實際上進行Selector注冊的時候我們有4種事件可以選擇監(jiān)聽:

  • SelectionKey.OP_CONNECT
  • SelectionKey.OP_ACCEPT
  • SelectionKey.OP_READ
  • SelectionKey.OP_WRITE

SelectionKey:
當(dāng)將一個Channel注冊到一個Selector上的時候 register() 方法會返回一個 SelectionKey對象肋层。該對象包括一些屬性:

  • interest Set
    代表你在注冊Channel的時候所感興趣的事件:
    Java
    int interestSet = selectionKey.interestOps();
    boolean isInterestedInAccept = interestSet & SelectionKey.OP_ACCEPT;
    boolean isInterestedInConnect = interestSet & SelectionKey.OP_CONNECT;
    boolean isInterestedInRead = interestSet & SelectionKey.OP_READ;
    boolean isInterestedInWrite = interestSet & SelectionKey.OP_WRITE;

- ReadySet
代表一個Channel已經(jīng)準(zhǔn)備好的操作集合:
```Java```
int readySet = selectionKey.readyOps();
selectionKey.isAcceptable();
selectionKey.isConnectable();
selectionKey.isReadable();
selectionKey.isWritable();
  • Channel + Selector
    通過SelectionKey訪問Channel和Selector是非常容易的:
    Java
    Channel channel = selectionKey.channel();
    Selector selector = selectionKey.selector();
- Attaching Objects
我們可以在SelectionKey上附加一個對象來儲存更多的信息亿笤。
```Java```
selectionKey.attach(theObject);
Object attachedObj = selectionKey.attachment();

通過Selector選取一個Channel
當(dāng)你為Selector注冊了一個或者多個Channel后你就可以通過select()方法選取已經(jīng)ready的Channels(注意s).select方法有以下三種:

  • int select() ; 如果當(dāng)前沒有已經(jīng)ready的Channel供選擇,該方法阻塞栋猖,知道某個Channel是ready狀態(tài)净薛,將它選取。
  • int select(long timeout) ; 超時時間timeout內(nèi)會進行等待蒲拉。
  • int selectNow() ; 不會阻塞肃拜,不管有沒有Channel準(zhǔn)備好,都會立即返回全陨。

select 方法返回的int值爆班,代表了當(dāng)前有多少個Channel是ready的。注意該值是指從你上一次調(diào)用select后辱姨,有多少個Channel變成了ready狀態(tài)。

//selectedKeys, to supplement

一個完整的selector工作示例:
Java
Selector selector = Selector.open();
channel.configureBlocking(false);
SelectionKey key = channel.register(selector, SelectionKey.OP_READ);

while(true) {
int readyChannels = selector.select();
if(readyChannels == 0) continue;
Set<SelectionKey> selectedKeys = selector.selectedKeys();
Iterator<SelectionKey> keyIterator = selectedKeys.iterator();
while(keyIterator.hasNext()) {
SelectionKey key = keyIterator.next();
if(key.isAcceptable()) {
// a connection was accepted by a ServerSocketChannel.
} else if (key.isConnectable()) {
// a connection was established with a remote server.
} else if (key.isReadable()) {
// a channel is ready for reading
} else if (key.isWritable()) {
// a channel is ready for writing
}
keyIterator.remove();
}
}

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末戚嗅,一起剝皮案震驚了整個濱河市雨涛,隨后出現(xiàn)的幾起案子枢舶,更是在濱河造成了極大的恐慌,老刑警劉巖替久,帶你破解...
    沈念sama閱讀 222,252評論 6 516
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件凉泄,死亡現(xiàn)場離奇詭異,居然都是意外死亡蚯根,警方通過查閱死者的電腦和手機后众,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,886評論 3 399
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來颅拦,“玉大人蒂誉,你說我怎么就攤上這事【嗨В” “怎么了右锨?”我有些...
    開封第一講書人閱讀 168,814評論 0 361
  • 文/不壞的土叔 我叫張陵,是天一觀的道長碌秸。 經(jīng)常有香客問我绍移,道長,這世上最難降的妖魔是什么讥电? 我笑而不...
    開封第一講書人閱讀 59,869評論 1 299
  • 正文 為了忘掉前任蹂窖,我火速辦了婚禮,結(jié)果婚禮上恩敌,老公的妹妹穿的比我還像新娘恼策。我一直安慰自己,他們只是感情好潮剪,可當(dāng)我...
    茶點故事閱讀 68,888評論 6 398
  • 文/花漫 我一把揭開白布涣楷。 她就那樣靜靜地躺著,像睡著了一般抗碰。 火紅的嫁衣襯著肌膚如雪狮斗。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 52,475評論 1 312
  • 那天弧蝇,我揣著相機與錄音碳褒,去河邊找鬼。 笑死看疗,一個胖子當(dāng)著我的面吹牛沙峻,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播两芳,決...
    沈念sama閱讀 41,010評論 3 422
  • 文/蒼蘭香墨 我猛地睜開眼摔寨,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了怖辆?” 一聲冷哼從身側(cè)響起是复,我...
    開封第一講書人閱讀 39,924評論 0 277
  • 序言:老撾萬榮一對情侶失蹤删顶,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后淑廊,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體逗余,經(jīng)...
    沈念sama閱讀 46,469評論 1 319
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 38,552評論 3 342
  • 正文 我和宋清朗相戀三年季惩,在試婚紗的時候發(fā)現(xiàn)自己被綠了录粱。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 40,680評論 1 353
  • 序言:一個原本活蹦亂跳的男人離奇死亡画拾,死狀恐怖啥繁,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情碾阁,我是刑警寧澤输虱,帶...
    沈念sama閱讀 36,362評論 5 351
  • 正文 年R本政府宣布,位于F島的核電站脂凶,受9級特大地震影響宪睹,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜蚕钦,卻給世界環(huán)境...
    茶點故事閱讀 42,037評論 3 335
  • 文/蒙蒙 一亭病、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧嘶居,春花似錦罪帖、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,519評論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至佑吝,卻和暖如春坐昙,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背芋忿。 一陣腳步聲響...
    開封第一講書人閱讀 33,621評論 1 274
  • 我被黑心中介騙來泰國打工炸客, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人戈钢。 一個月前我還...
    沈念sama閱讀 49,099評論 3 378
  • 正文 我出身青樓痹仙,卻偏偏與公主長得像,于是被迫代替她去往敵國和親殉了。 傳聞我的和親對象是個殘疾皇子开仰,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 45,691評論 2 361

推薦閱讀更多精彩內(nèi)容

  • Java NIO(New IO)是從Java 1.4版本開始引入的一個新的IO API,可以替代標(biāo)準(zhǔn)的Java I...
    JackChen1024閱讀 7,557評論 1 143
  • 簡介 Java NIO 是由 Java 1.4 引進的異步 IO.Java NIO 由以下幾個核心部分組成: Ch...
    永順閱讀 1,794評論 0 15
  • (轉(zhuǎn)載說明:本文非原創(chuàng),轉(zhuǎn)載自http://ifeve.com/java-nio-all/) Java NIO: ...
    數(shù)獨題閱讀 810評論 0 3
  • 前言: 之前的文章《Java文件IO常用歸納》主要寫了Java 標(biāo)準(zhǔn)IO要注意的細節(jié)和技巧抖所,由于網(wǎng)上各種學(xué)習(xí)途徑梨州,...
    androidjp閱讀 2,909評論 0 22
  • (轉(zhuǎn)載說明:本文非原創(chuàng),轉(zhuǎn)載自http://ifeve.com/java-nio-all/) Java NIO: ...
    柳岸閱讀 822評論 0 3