摘要:抽象類有一個(gè)方法用于使通道處于阻塞模式或非阻塞模式。注意抽象類的方法是由抽象類實(shí)現(xiàn)的,都是直接繼承了抽象類。大家有興趣可以看看的源碼,各種抽象類和抽象類上層的抽象類。
歷史回顧:
Java NIO 概覽
Java NIO 之 Buffer(緩沖區(qū))
Java NIO 之 Channel(通道)
其他高贊文章:
面試中關(guān)于Redis的問(wèn)題看這篇就夠了
一文輕松搞懂redis集群原理及搭建與使用
超詳細(xì)的Java面試題總結(jié)(三)之Java集合篇常見問(wèn)題
一 Selector(選擇器)介紹Selector 一般稱 為選擇器 ,當(dāng)然你也可以翻譯為 多路復(fù)用器 。它是Java NIO核心組件中的一個(gè),用于檢查一個(gè)或多個(gè)NIO Channel(通道)的狀態(tài)是否處于可讀、可寫。如此可以實(shí)現(xiàn)單線程管理多個(gè)channels,也就是可以管理多個(gè)網(wǎng)絡(luò)鏈接。
使用Selector的好處在于: 使用更少的線程來(lái)就可以來(lái)處理通道了, 相比使用多個(gè)線程,避免了線程上下文切換帶來(lái)的開銷。
二 Selector(選擇器)的使用方法介紹 1. Selector的創(chuàng)建通過(guò)調(diào)用Selector.open()方法創(chuàng)建一個(gè)Selector對(duì)象,如下:
Selector selector = Selector.open();
這里需要說(shuō)明一下
2. 注冊(cè)Channel到Selectorchannel.configureBlocking(false); SelectionKey key = channel.register(selector, Selectionkey.OP_READ);
Channel必須是非阻塞的。
所以FileChannel不適用Selector,因?yàn)镕ileChannel不能切換為非阻塞模式,更準(zhǔn)確的來(lái)說(shuō)是因?yàn)镕ileChannel沒(méi)有繼承SelectableChannel。Socket channel可以正常使用。
SelectableChannel抽象類 有一個(gè) configureBlocking() 方法用于使通道處于阻塞模式或非阻塞模式。
abstract SelectableChannel configureBlocking(boolean block)注意:
SelectableChannel抽象類的configureBlocking() 方法是由 AbstractSelectableChannel抽象類實(shí)現(xiàn)的,SocketChannel、ServerSocketChannel、DatagramChannel都是直接繼承了 AbstractSelectableChannel抽象類 。
大家有興趣可以看看NIO的源碼,各種抽象類和抽象類上層的抽象類。我本人暫時(shí)不準(zhǔn)備研究NIO源碼,因?yàn)檫€有很多事情要做,需要研究的同學(xué)可以自行看看。
register() 方法的第二個(gè)參數(shù)。這是一個(gè)“ interest集合 ”,意思是在通過(guò)Selector監(jiān)聽Channel時(shí)對(duì)什么事件感興趣??梢员O(jiān)聽四種不同類型的事件:
Connect
Accept
Read
Write
通道觸發(fā)了一個(gè)事件意思是該事件已經(jīng)就緒。比如某個(gè)Channel成功連接到另一個(gè)服務(wù)器稱為“ 連接就緒 ”。一個(gè)Server Socket Channel準(zhǔn)備好接收新進(jìn)入的連接稱為“ 接收就緒 ”。一個(gè)有數(shù)據(jù)可讀的通道可以說(shuō)是“ 讀就緒 ”。等待寫數(shù)據(jù)的通道可以說(shuō)是“ 寫就緒 ”。
這四種事件用SelectionKey的四個(gè)常量來(lái)表示:
SelectionKey.OP_CONNECT SelectionKey.OP_ACCEPT SelectionKey.OP_READ SelectionKey.OP_WRITE
如果你對(duì)不止一種事件感興趣,使用或運(yùn)算符即可,如下:
int interestSet = SelectionKey.OP_READ | SelectionKey.OP_WRITE;3. SelectionKey介紹
一個(gè)SelectionKey鍵表示了一個(gè)特定的通道對(duì)象和一個(gè)特定的選擇器對(duì)象之間的注冊(cè)關(guān)系。
key.attachment(); //返回SelectionKey的attachment,attachment可以在注冊(cè)channel的時(shí)候指定。 key.channel(); // 返回該SelectionKey對(duì)應(yīng)的channel。 key.selector(); // 返回該SelectionKey對(duì)應(yīng)的Selector。 key.interestOps(); //返回代表需要Selector監(jiān)控的IO操作的bit mask key.readyOps(); // 返回一個(gè)bit mask,代表在相應(yīng)channel上可以進(jìn)行的IO操作。
key.interestOps():
我們可以通過(guò)以下方法來(lái)判斷Selector是否對(duì)Channel的某種事件感興趣
int interestSet = selectionKey.interestOps(); boolean isInterestedInAccept = (interestSet & SelectionKey.OP_ACCEPT) == SelectionKey.OP_ACCEPT; boolean isInterestedInConnect = interestSet & SelectionKey.OP_CONNECT; boolean isInterestedInRead = interestSet & SelectionKey.OP_READ; boolean isInterestedInWrite = interestSet & SelectionKey.OP_WRITE;
key.readyOps()
ready 集合是通道已經(jīng)準(zhǔn)備就緒的操作的集合。JAVA中定義以下幾個(gè)方法用來(lái)檢查這些操作是否就緒.
//創(chuàng)建ready集合的方法 int readySet = selectionKey.readyOps(); //檢查這些操作是否就緒的方法 key.isAcceptable();//是否可讀,是返回 true boolean isWritable()://是否可寫,是返回 true boolean isConnectable()://是否可連接,是返回 true boolean isAcceptable()://是否可接收,是返回 true
從SelectionKey訪問(wèn)Channel和Selector很簡(jiǎn)單。如下:
Channel channel = key.channel(); Selector selector = key.selector(); key.attachment();
可以將一個(gè)對(duì)象或者更多信息附著到SelectionKey上,這樣就能方便的識(shí)別某個(gè)給定的通道。例如,可以附加 與通道一起使用的Buffer,或是包含聚集數(shù)據(jù)的某個(gè)對(duì)象。使用方法如下:
key.attach(theObject); Object attachedObj = key.attachment();
還可以在用register()方法向Selector注冊(cè)Channel的時(shí)候附加對(duì)象。如:
SelectionKey key = channel.register(selector, SelectionKey.OP_READ, theObject);4. 從Selector中選擇channel(Selecting Channels via a Selector)
選擇器維護(hù)注冊(cè)過(guò)的通道的集合,并且這種注冊(cè)關(guān)系都被封裝在SelectionKey當(dāng)中.
Selector維護(hù)的三種類型SelectionKey集合:
已注冊(cè)的鍵的集合(Registered key set)
所有與選擇器關(guān)聯(lián)的通道所生成的鍵的集合稱為已經(jīng)注冊(cè)的鍵的集合。并不是所有注冊(cè)過(guò)的鍵都仍然有效。這個(gè)集合通過(guò) keys() 方法返回,并且可能是空的。這個(gè)已注冊(cè)的鍵的集合不是可以直接修改的;試圖這么做的話將引發(fā)java.lang.UnsupportedOperationException。
已選擇的鍵的集合(Selected key set)
所有與選擇器關(guān)聯(lián)的通道所生成的鍵的集合稱為已經(jīng)注冊(cè)的鍵的集合。并不是所有注冊(cè)過(guò)的鍵都仍然有效。這個(gè)集合通過(guò) keys() 方法返回,并且可能是空的。這個(gè)已注冊(cè)的鍵的集合不是可以直接修改的;試圖這么做的話將引發(fā)java.lang.UnsupportedOperationException。
已取消的鍵的集合(Cancelled key set)
已注冊(cè)的鍵的集合的子集,這個(gè)集合包含了 cancel() 方法被調(diào)用過(guò)的鍵(這個(gè)鍵已經(jīng)被無(wú)效化),但它們還沒(méi)有被注銷。這個(gè)集合是選擇器對(duì)象的私有成員,因而無(wú)法直接訪問(wèn)。
注意:
當(dāng)鍵被取消( 可以通過(guò)isValid( ) 方法來(lái)判斷)時(shí),它將被放在相關(guān)的選擇器的已取消的鍵的集合里。注冊(cè)不會(huì)立即被取消,但鍵會(huì)立即失效。當(dāng)再次調(diào)用 select( ) 方法時(shí)(或者一個(gè)正在進(jìn)行的select()調(diào)用結(jié)束時(shí)),已取消的鍵的集合中的被取消的鍵將被清理掉,并且相應(yīng)的注銷也將完成。通道會(huì)被注銷,而新的SelectionKey將被返回。當(dāng)通道關(guān)閉時(shí),所有相關(guān)的鍵會(huì)自動(dòng)取消(記住,一個(gè)通道可以被注冊(cè)到多個(gè)選擇器上)。當(dāng)選擇器關(guān)閉時(shí),所有被注冊(cè)到該選擇器的通道都將被注銷,并且相關(guān)的鍵將立即被無(wú)效化(取消)。一旦鍵被無(wú)效化,調(diào)用它的與選擇相關(guān)的方法就將拋出CancelledKeyException。
select()方法介紹:
在剛初始化的Selector對(duì)象中,這三個(gè)集合都是空的。 通過(guò)Selector的select()方法可以選擇已經(jīng)準(zhǔn)備就緒的通道 (這些通道包含你感興趣的的事件)。比如你對(duì)讀就緒的通道感興趣,那么select()方法就會(huì)返回讀事件已經(jīng)就緒的那些通道。下面是Selector幾個(gè)重載的select()方法:
int select():阻塞到至少有一個(gè)通道在你注冊(cè)的事件上就緒了。
int select(long timeout):和select()一樣,但最長(zhǎng)阻塞時(shí)間為timeout毫秒。
int selectNow():非阻塞,只要有通道就緒就立刻返回。
select()方法返回的int值表示有多少通道已經(jīng)就緒,是自上次調(diào)用select()方法后有多少通道變成就緒狀態(tài)。之前在select()調(diào)用時(shí)進(jìn)入就緒的通道不會(huì)在本次調(diào)用中被記入,而在前一次select()調(diào)用進(jìn)入就緒但現(xiàn)在已經(jīng)不在處于就緒的通道也不會(huì)被記入。例如:首次調(diào)用select()方法,如果有一個(gè)通道變成就緒狀態(tài),返回了1,若再次調(diào)用select()方法,如果另一個(gè)通道就緒了,它會(huì)再次返回1。如果對(duì)第一個(gè)就緒的channel沒(méi)有做任何操作,現(xiàn)在就有兩個(gè)就緒的通道,但在每次select()方法調(diào)用之間,只有一個(gè)通道就緒了。
一旦調(diào)用select()方法,并且返回值不為0時(shí),則 可以通過(guò)調(diào)用Selector的selectedKeys()方法來(lái)訪問(wèn)已選擇鍵集合 。如下:
Set selectedKeys=selector.selectedKeys();
進(jìn)而可以放到和某SelectionKey關(guān)聯(lián)的Selector和Channel。如下所示:
Set selectedKeys = selector.selectedKeys(); Iterator 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(); }5. 停止選擇的方法
選擇器執(zhí)行選擇的過(guò)程,系統(tǒng)底層會(huì)依次詢問(wèn)每個(gè)通道是否已經(jīng)就緒,這個(gè)過(guò)程可能會(huì)造成調(diào)用線程進(jìn)入阻塞狀態(tài),那么我們有以下三種方式可以喚醒在select()方法中阻塞的線程。
wakeup()方法 :通過(guò)調(diào)用Selector對(duì)象的wakeup()方法讓處在阻塞狀態(tài)的select()方法立刻返回
該方法使得選擇器上的第一個(gè)還沒(méi)有返回的選擇操作立即返回。如果當(dāng)前沒(méi)有進(jìn)行中的選擇操作,那么下一次對(duì)select()方法的一次調(diào)用將立即返回。
close()方法 :通過(guò)close()方法關(guān)閉Selector,
該方法使得任何一個(gè)在選擇操作中阻塞的線程都被喚醒(類似wakeup()),同時(shí)使得注冊(cè)到該Selector的所有Channel被注銷,所有的鍵將被取消,但是Channel本身并不會(huì)關(guān)閉。
三 模板代碼一個(gè)服務(wù)端的模板代碼:
有了模板代碼我們?cè)诰帉懗绦驎r(shí),大多數(shù)時(shí)間都是在模板代碼中添加相應(yīng)的業(yè)務(wù)代碼
ServerSocketChannel ssc = ServerSocketChannel.open(); ssc.socket().bind(new InetSocketAddress("localhost", 8080)); ssc.configureBlocking(false); Selector selector = Selector.open(); ssc.register(selector, SelectionKey.OP_ACCEPT); while(true) { int readyNum = selector.select(); if (readyNum == 0) { continue; } Set四 客戶端與服務(wù)端簡(jiǎn)單交互實(shí)例selectedKeys = selector.selectedKeys(); Iterator it = selectedKeys.iterator(); while(it.hasNext()) { SelectionKey key = it.next(); if(key.isAcceptable()) { // 接受連接 } else if (key.isReadable()) { // 通道可讀 } else if (key.isWritable()) { // 通道可寫 } it.remove(); } }
服務(wù)端:
package selector; import java.io.IOException; import java.net.InetSocketAddress; import java.nio.ByteBuffer; import java.nio.channels.SelectionKey; import java.nio.channels.Selector; import java.nio.channels.ServerSocketChannel; import java.nio.channels.SocketChannel; import java.util.Iterator; import java.util.Set; public class WebServer { public static void main(String[] args) { try { ServerSocketChannel ssc = ServerSocketChannel.open(); ssc.socket().bind(new InetSocketAddress("127.0.0.1", 8000)); ssc.configureBlocking(false); Selector selector = Selector.open(); // 注冊(cè) channel,并且指定感興趣的事件是 Accept ssc.register(selector, SelectionKey.OP_ACCEPT); ByteBuffer readBuff = ByteBuffer.allocate(1024); ByteBuffer writeBuff = ByteBuffer.allocate(128); writeBuff.put("received".getBytes()); writeBuff.flip(); while (true) { int nReady = selector.select(); Setkeys = selector.selectedKeys(); Iterator it = keys.iterator(); while (it.hasNext()) { SelectionKey key = it.next(); it.remove(); if (key.isAcceptable()) { // 創(chuàng)建新的連接,并且把連接注冊(cè)到selector上,而且, // 聲明這個(gè)channel只對(duì)讀操作感興趣。 SocketChannel socketChannel = ssc.accept(); socketChannel.configureBlocking(false); socketChannel.register(selector, SelectionKey.OP_READ); } else if (key.isReadable()) { SocketChannel socketChannel = (SocketChannel) key.channel(); readBuff.clear(); socketChannel.read(readBuff); readBuff.flip(); System.out.println("received : " + new String(readBuff.array())); key.interestOps(SelectionKey.OP_WRITE); } else if (key.isWritable()) { writeBuff.rewind(); SocketChannel socketChannel = (SocketChannel) key.channel(); socketChannel.write(writeBuff); key.interestOps(SelectionKey.OP_READ); } } } } catch (IOException e) { e.printStackTrace(); } } }
客戶端:
package selector; import java.io.IOException; import java.net.InetSocketAddress; import java.nio.ByteBuffer; import java.nio.channels.SocketChannel; public class WebClient { public static void main(String[] args) throws IOException { try { SocketChannel socketChannel = SocketChannel.open(); socketChannel.connect(new InetSocketAddress("127.0.0.1", 8000)); ByteBuffer writeBuffer = ByteBuffer.allocate(32); ByteBuffer readBuffer = ByteBuffer.allocate(32); writeBuffer.put("hello".getBytes()); writeBuffer.flip(); while (true) { writeBuffer.rewind(); socketChannel.write(writeBuffer); readBuffer.clear(); socketChannel.read(readBuffer); } } catch (IOException e) { } } }
運(yùn)行結(jié)果:
先運(yùn)行服務(wù)端,再運(yùn)行客戶端,服務(wù)端會(huì)不斷收到客戶端發(fā)送過(guò)來(lái)的消息。
其他實(shí)例:
《基于 Java NIO 實(shí)現(xiàn)簡(jiǎn)單的 HTTP 服務(wù)器》
參考:官方JDK相關(guān)文檔
谷歌搜索排名第一的Java NIO教程
《Java NIO》
《Java 8編程官方參考教程(第9版)》
Java NIO Selector詳解(含多人聊天室實(shí)例)
Java NIO(6): Selector
歡迎關(guān)注我的微信公眾號(hào):"Java面試通關(guān)手冊(cè)"(一個(gè)有溫度的微信公眾號(hào),期待與你共同進(jìn)步~~~堅(jiān)持原創(chuàng),分享美文,分享各種Java學(xué)習(xí)資源):
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://m.hztianpu.com/yun/69404.html
摘要:從通道進(jìn)行數(shù)據(jù)寫入創(chuàng)建一個(gè)緩沖區(qū),填充數(shù)據(jù),并要求通道寫入數(shù)據(jù)。三之通道主要內(nèi)容通道介紹通常來(lái)說(shuō)中的所有都是從通道開始的。從中選擇選擇器維護(hù)注冊(cè)過(guò)的通道的集合,并且這種注冊(cè)關(guān)系都被封裝在當(dāng)中停止選擇的方法方法和方法。 由于內(nèi)容比較多,我下面放的一部分是我更新在我的微信公眾號(hào)上的鏈接,微信排版比較好看,更加利于閱讀。每一篇文章下面我都把文章的主要內(nèi)容給列出來(lái)了,便于大家學(xué)習(xí)與回顧。 Ja...
摘要:學(xué)習(xí)和掌握技術(shù)已經(jīng)不是一個(gè)攻城獅的加分技能,而是一個(gè)必備技能。是雙向的,不僅可以讀取數(shù)據(jù)還能保存數(shù)據(jù),程序不能直接讀寫通道,只與緩沖區(qū)交互為了讓大家不被高并發(fā)與大量連接處理問(wèn)題所困擾,動(dòng)力節(jié)點(diǎn)推出了高效處理模型應(yīng)用教程。 大家肯定了解Java IO, 但是對(duì)于NIO一般是陌生的,而現(xiàn)在使用到NIO的場(chǎng)景越來(lái)越多,很多技術(shù)框...
摘要:線程之間的切換對(duì)于操作系統(tǒng)來(lái)說(shuō)是昂貴的。因此,單線程可以監(jiān)視多個(gè)通道中的數(shù)據(jù)。當(dāng)方法返回后,線程可以處理這些事件。 一 NIO簡(jiǎn)介 Java NIO 是 java 1.4 之后新出的一套IO接口,這里的的新是相對(duì)于原有標(biāo)準(zhǔn)的Java IO和Java Networking接口。NIO提供了一種完全不同的操作方式。 NIO中的N可以理解為Non-blocking,不單純是New。 它支持面...
摘要:編程核心是通道和選擇器,選擇器通過(guò)不斷輪詢,執(zhí)行對(duì)應(yīng)的函數(shù)。所以我們需要捕獲這個(gè)異常,并且開始不斷重連。如果客戶端關(guān)閉那么服務(wù)器也要主動(dòng)關(guān)閉他數(shù)據(jù)庫(kù)代碼及實(shí)體類如果還想實(shí)現(xiàn)數(shù)據(jù)庫(kù)方面代碼,私我 ...
摘要:組件主要有三大核心部分通道,緩沖區(qū)選擇器。選擇區(qū)用于監(jiān)聽多個(gè)通道的事件比如連接打開,數(shù)據(jù)到達(dá)。即用選擇器,借助單一線程,就可對(duì)數(shù)量龐大的活動(dòng)通道實(shí)施監(jiān)控和維護(hù)。 Java NIO 簡(jiǎn)介JAVA NIO有兩種解釋:一種叫非阻塞IO(Non-blocking I/O),另一種也叫新的IO(New I/O),其實(shí)是同一個(gè)概念。它是一種同步非阻塞的I/O模型,也是I/O多路復(fù)用的基礎(chǔ),已經(jīng)被越...
閱讀 2562·2023-04-25 19:24
閱讀 1789·2021-11-11 16:54
閱讀 2893·2021-11-08 13:19
閱讀 3617·2021-10-25 09:45
閱讀 2629·2021-09-13 10:24
閱讀 3394·2021-09-07 10:15
閱讀 4201·2021-09-07 10:14
閱讀 3024·2019-08-30 15:56