歷史回顧:html
Java NIO 概覽java
其餘高贊文章:編程
超詳細的Java面試題總結(三)之Java集合篇常見問題服務器
Selector 通常稱 爲選擇器 ,固然你也能夠翻譯爲 多路複用器 。它是Java NIO核心組件中的一個,用於檢查一個或多個NIO Channel(通道)的狀態是否處於可讀、可寫。如此能夠實現單線程管理多個channels,也就是能夠管理多個網絡連接。微信
使用Selector的好處在於: 使用更少的線程來就能夠來處理通道了, 相比使用多個線程,避免了線程上下文切換帶來的開銷。網絡
經過調用Selector.open()方法建立一個Selector對象,以下:
Selector selector = Selector.open();
這裏須要說明一下
channel.configureBlocking(false);
SelectionKey key = channel.register(selector, Selectionkey.OP_READ);
Channel必須是非阻塞的。
因此FileChannel不適用Selector,由於FileChannel不能切換爲非阻塞模式,更準確的來講是由於FileChannel沒有繼承SelectableChannel。Socket channel能夠正常使用。
SelectableChannel抽象類 有一個 configureBlocking() 方法用於使通道處於阻塞模式或非阻塞模式。
abstract SelectableChannel configureBlocking(boolean block)
SelectableChannel抽象類的configureBlocking() 方法是由 AbstractSelectableChannel抽象類實現的,SocketChannel、ServerSocketChannel、DatagramChannel都是直接繼承了 AbstractSelectableChannel抽象類 。
你們有興趣能夠看看NIO的源碼,各類抽象類和抽象類上層的抽象類。我本人暫時不許備研究NIO源碼,由於還有不少事情要作,須要研究的同窗能夠自行看看。
register() 方法的第二個參數。這是一個「 interest集合 」,意思是在經過Selector監聽Channel時對什麼事件感興趣。能夠監聽四種不一樣類型的事件:
通道觸發了一個事件意思是該事件已經就緒。好比某個Channel成功鏈接到另外一個服務器稱爲「 鏈接就緒 」。一個Server Socket Channel準備好接收新進入的鏈接稱爲「 接收就緒 」。一個有數據可讀的通道能夠說是「 讀就緒 」。等待寫數據的通道能夠說是「 寫就緒 」。
這四種事件用SelectionKey的四個常量來表示:
SelectionKey.OP_CONNECT SelectionKey.OP_ACCEPT SelectionKey.OP_READ SelectionKey.OP_WRITE
若是你對不止一種事件感興趣,使用或運算符便可,以下:
int interestSet = SelectionKey.OP_READ | SelectionKey.OP_WRITE;
一個SelectionKey鍵表示了一個特定的通道對象和一個特定的選擇器對象之間的註冊關係。
key.attachment(); //返回SelectionKey的attachment,attachment能夠在註冊channel的時候指定。
key.channel(); // 返回該SelectionKey對應的channel。
key.selector(); // 返回該SelectionKey對應的Selector。
key.interestOps(); //返回表明須要Selector監控的IO操做的bit mask
key.readyOps(); // 返回一個bit mask,表明在相應channel上能夠進行的IO操做。
key.interestOps():
咱們能夠經過如下方法來判斷Selector是否對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 集合是通道已經準備就緒的操做的集合。JAVA中定義如下幾個方法用來檢查這些操做是否就緒.
//建立ready集合的方法
int readySet = selectionKey.readyOps();
//檢查這些操做是否就緒的方法
key.isAcceptable();//是否可讀,是返回 true
boolean isWritable()://是否可寫,是返回 true
boolean isConnectable()://是否可鏈接,是返回 true
boolean isAcceptable()://是否可接收,是返回 true
從SelectionKey訪問Channel和Selector很簡單。以下:
Channel channel = key.channel(); Selector selector = key.selector(); key.attachment();
能夠將一個對象或者更多信息附着到SelectionKey上,這樣就能方便的識別某個給定的通道。例如,能夠附加 與通道一塊兒使用的Buffer,或是包含彙集數據的某個對象。使用方法以下:
key.attach(theObject); Object attachedObj = key.attachment();
還能夠在用register()方法向Selector註冊Channel的時候附加對象。如:
SelectionKey key = channel.register(selector, SelectionKey.OP_READ, theObject);
選擇器維護註冊過的通道的集合,而且這種註冊關係都被封裝在SelectionKey當中.
Selector維護的三種類型SelectionKey集合:
已註冊的鍵的集合(Registered key set)
全部與選擇器關聯的通道所生成的鍵的集合稱爲已經註冊的鍵的集合。並非全部註冊過的鍵都仍然有效。這個集合經過 keys() 方法返回,而且多是空的。這個已註冊的鍵的集合不是能夠直接修改的;試圖這麼作的話將引起java.lang.UnsupportedOperationException。
已選擇的鍵的集合(Selected key set)
全部與選擇器關聯的通道所生成的鍵的集合稱爲已經註冊的鍵的集合。並非全部註冊過的鍵都仍然有效。這個集合經過 keys() 方法返回,而且多是空的。這個已註冊的鍵的集合不是能夠直接修改的;試圖這麼作的話將引起java.lang.UnsupportedOperationException。
已取消的鍵的集合(Cancelled key set)
已註冊的鍵的集合的子集,這個集合包含了 cancel() 方法被調用過的鍵(這個鍵已經被無效化),但它們尚未被註銷。這個集合是選擇器對象的私有成員,於是沒法直接訪問。
注意:
當鍵被取消( 能夠經過isValid( ) 方法來判斷)時,它將被放在相關的選擇器的已取消的鍵的集合裏。註冊不會當即被取消,但鍵會當即失效。當再次調用 select( ) 方法時(或者一個正在進行的select()調用結束時),已取消的鍵的集合中的被取消的鍵將被清理掉,而且相應的註銷也將完成。通道會被註銷,而新的SelectionKey將被返回。當通道關閉時,全部相關的鍵會自動取消(記住,一個通道能夠被註冊到多個選擇器上)。當選擇器關閉時,全部被註冊到該選擇器的通道都將被註銷,而且相關的鍵將當即被無效化(取消)。一旦鍵被無效化,調用它的與選擇相關的方法就將拋出CancelledKeyException。
select()方法介紹:
在剛初始化的Selector對象中,這三個集合都是空的。 經過Selector的select()方法能夠選擇已經準備就緒的通道 (這些通道包含你感興趣的的事件)。好比你對讀就緒的通道感興趣,那麼select()方法就會返回讀事件已經就緒的那些通道。下面是Selector幾個重載的select()方法:
select()方法返回的int值表示有多少通道已經就緒,是自上次調用select()方法後有多少通道變成就緒狀態。以前在select()調用時進入就緒的通道不會在本次調用中被記入,而在前一次select()調用進入就緒但如今已經不在處於就緒的通道也不會被記入。例如:首次調用select()方法,若是有一個通道變成就緒狀態,返回了1,若再次調用select()方法,若是另外一個通道就緒了,它會再次返回1。若是對第一個就緒的channel沒有作任何操做,如今就有兩個就緒的通道,但在每次select()方法調用之間,只有一個通道就緒了。
一旦調用select()方法,而且返回值不爲0時,則 能夠經過調用Selector的selectedKeys()方法來訪問已選擇鍵集合 。以下:
Set selectedKeys=selector.selectedKeys();
進而能夠放到和某SelectionKey關聯的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();
}
選擇器執行選擇的過程,系統底層會依次詢問每一個通道是否已經就緒,這個過程可能會形成調用線程進入阻塞狀態,那麼咱們有如下三種方式能夠喚醒在select()方法中阻塞的線程。
一個服務端的模板代碼:
有了模板代碼咱們在編寫程序時,大多數時間都是在模板代碼中添加相應的業務代碼
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<SelectionKey> selectedKeys = selector.selectedKeys();
Iterator<SelectionKey> it = selectedKeys.iterator();
while(it.hasNext()) {
SelectionKey key = it.next();
if(key.isAcceptable()) {
// 接受鏈接
} else if (key.isReadable()) {
// 通道可讀
} else if (key.isWritable()) {
// 通道可寫
}
it.remove();
}
}
服務端:
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();
// 註冊 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();
Set<SelectionKey> keys = selector.selectedKeys();
Iterator<SelectionKey> it = keys.iterator();
while (it.hasNext()) {
SelectionKey key = it.next();
it.remove();
if (key.isAcceptable()) {
// 建立新的鏈接,而且把鏈接註冊到selector上,並且,
// 聲明這個channel只對讀操做感興趣。
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) {
}
}
}
運行結果:
先運行服務端,再運行客戶端,服務端會不斷收到客戶端發送過來的消息。
其餘實例:
歡迎關注個人微信公衆號:」Java面試通關手冊」(一個有溫度的微信公衆號,期待與你共同進步~~~堅持原創,分享美文,分享各類Java學習資源):