前面的两篇文章中总结了Java NIO中的两大基础组件Buffer和Channel的相关知识点,在NIO中都是通过Channel和Buffer的协作来读写数据的,在这个基础上通过selector来协调多个channel以同时读写数据,本文我们就来学习一下selector。

  Java NIO中引入了"selector"的概念,一个selector其实是一个Java对象,能够通过诸如连接打开、数据就绪等事件监控多个channel。如此在单个线程中就可以通过一个selector同时处理多个channel,同样也可以同时处理多个网络连接。

  本文会围绕如下几个方面展开:

  为什么要有selector

  创建selector及注册channel

  SelectionKey

  从Selector中选择Channels  

  Selector的一些其他操作

  总结

1. 为什么要有selector

  利用单个线程处理多个channel的好处是可以减少线程的数量,节省开销。实际上,可以只用一个线程处理所有的channels。因为线程间的切换是很耗费操作系统资源的一项操作,并且每个线程都要占用一定操作系统资源(内存)。因此,线程数量当然是越少越好,而通过引入selector的概念可以显著减少线程的数量,同时又不会减少系统处理的连接数,可以简单理解为吞吐量。

  如下示例解释了一个Selector处理3个Channel:

2. 创建selector及注册channel

  通过调用Selector的静态方法open()来创建一个selector对象:

Selector selector = Selector.open();

  要让Selector处理Channel就需要先将Channel注册到Selector中,可以通过调用SelectableChannel.register()方法完成:

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

  Channel必须处于非阻塞模式下才能被Selector所使用,因此FileChannel不能为Selector所用,因为它不能切换到非阻塞模式。Socket channels则支持这种工作模式,通过channel的configureBlocking()方法设置其工作模式是阻塞还是非阻塞式。

  register()的第二个参数是一个set集合(interest set),用来指代那些Selector有兴趣监听的事件。一共有四种事件类型:

  • Connect
  • Accept
  • Read
  • Write

  一个channel触发了某一事件其实是代表着它的某些状态已经就绪,四种事件类型分别对应如下四种情况:

  • 一个channel成功和服务器连接上就是"connect ready",由SelectionKey.OP_CONNECT指代;
  • 一个server socket channel接受了一个连接就称为"accept ready",由SelectionKey.OP_ACCEPT指代;
  • 一个channel中数据准备好了被读就是"read ready",由SelectionKey.OP_READ指代;
  • 一个channel可供写入数据就是"write ready",由SelectionKey.OP_WRITE指代;

  通过这种传参的方式,我们可以指定selector监听channel哪些事件,如果需要同时表示多种事件,则可以如下方式来表示:

int interestSet = SelectionKey.OP_READ | SelectionKey.OP_WRITE; 

3. SelectionKey

  如上一部分所示,当往Selector中注册一个Channel时,register()方法会返回一个SelectionKey对象,其包含了如下属性:

  • The interest set
  • The ready set
  • The Channel
  • The Selector
  • An attached object (optional)

3.1 Interest Set

  可以通过如下方法读写Interest Set:

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;

  可以看到,采用的是与的方式来判断一个指定的event是否在interestSet中

3.2 Ready Set

  这个指代channel就绪的操作的集合。在selection之后就能够获得一个ready set(这个selection稍后会介绍到),可以通过如下方式获取:

int readySet = selectionKey.readyOps();

  可以通过如下方式判断是否就绪:

selectionKey.isAcceptable();
selectionKey.isConnectable();
selectionKey.isReadable();
selectionKey.isWritable();

3.3 Channel + Selector

  通过如下方式来获取channel和selector:

Channel  channel  = selectionKey.channel();
Selector selector = selectionKey.selector();

3.4 Attaching Objects

  可以给SelectionKey附带对象,这是一个手动标记一个channel的方式,或者是给channel附带更多信息的方式。你可以附带和channel连接的Buffer或者别的对象,使用方式如下:

// 可以这样搞
selectionKey.attach(theObject);
Object attachedObj = selectionKey.attachment(); // 也可以这样搞
SelectionKey key = channel.register(selector, SelectionKey.OP_READ, theObject);

4. 从Selector中选择Channels

  当你往Selector中注册了多个channel时,你可以调用select()方法用以获取感兴趣且就绪的channel,该方法有如下三种重载格式:

int select()
int select(long timeout)
int selectNow()
  • select()会一直阻塞,直到有一个channel就绪;
  • select(long timeout)只会阻塞一段指定的时间(单位为ms),直到有channel就绪;
  • selectNow()不会阻塞,不管是否有channel就绪都会立即返回;

  返回的int值指代有多少channel就绪(是从上一次调用select()之后开始算起)。比如调用select(),返回1,再次调用select(),这时又有一个channel就绪,此时任然是返回1。

4.1 selectedKeys()

  当调用了一次select()方法并且返回一个int值,这时你可以通过"selected key set"来获取这些就绪的channels:

Set<SelectionKey> selectedKeys = selector.selectedKeys(); 

  通过调用selectedKeys()方法,返回的是一个Set,你可以遍历以获取就绪的channel:

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();
}

  通过判断其对应的事件类型来做对应的操作。

5. Selector的一些其他操作

5.1 wakeUp()

  线程调用Selector的select()方法之后会阻塞,在这种情况下可以通过另一个线程调用同一个Selector的wakeup()方法来将其唤醒。如果一个线程调用了Selector的wakeup方法但是当前并没有线程阻塞,则下一个调用Selector的select()方法的线程则不会阻塞(还记得不,前面讲到select()方法会一直阻塞)。

5.2 close()

  当使用完了Selector,需要调用其close()方法来释放资源,该方法会关闭Selector并使所有相关的SelectionKey失效,但是和Selector相关的channel并不会被关闭。

5.3 一个例子

  开启一个Selector,往其中注册一个channel,并且一直监控:

Selector selector = Selector.open();
channel.configureBlocking(false);
SelectionKey key = channel.register(selector, SelectionKey.OP_READ);
while(true) {
int readyChannels = selector.selectNow();
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();
}
}

6. 总结

  本文简单总结了Java NIO中的Selector,有了Selector,我们可以实现多路复用,通过少量的线程来监控大量的连接,实现更高性能的服务器,很多现代服务器中都采用了这一特性,比如Tomcat、netty。

  我们可以往Selector中注册一些Channel,并且指定我们需要监听的事件类型,然后监控这些channel,一旦获取到就绪的事件,则可以执行下一部的操作,这就是一个Selector处理channel的基本流程。

Java NIO学习系列三:Selector的更多相关文章

  1. Java NIO学习系列一:Buffer

    前面三篇文章中分别总结了标准Java IO系统中的File.RandomAccessFile.I/O流系统,对于I/O系统从其继承体系入手,力求对类数量繁多的的I/O系统有一个清晰的认识,然后结合一些 ...

  2. Java NIO学习系列四:NIO和IO对比

    前面的一些文章中我总结了一些Java IO和NIO相关的主要知识点,也是管中窥豹,IO类库已经功能很强大了,但是Java 为什么又要引入NIO,这是我一直不是很清楚的?前面也只是简单提及了一下:因为性 ...

  3. Java NIO学习系列二:Channel

    上文总结了Java NIO中的Buffer相关知识点,本文中我们来总结一下它的好兄弟:Channel.上文有说到,Java NIO中的Buffer一般和Channel配对使用,NIO中的所有IO都起始 ...

  4. Java NIO学习系列六:Java中的IO模型

    前文中我们总结了linux系统中的5中IO模型,并且着重介绍了其中的4种IO模型: 阻塞I/O(blocking IO) 非阻塞I/O(nonblocking IO) I/O多路复用(IO multi ...

  5. Java NIO学习系列七:Path、Files、AsynchronousFileChannel

    相对于标准Java IO中通过File来指向文件和目录,Java NIO中提供了更丰富的类来支持对文件和目录的操作,不仅仅支持更多操作,还支持诸如异步读写等特性,本文我们就来学习一些Java NIO提 ...

  6. Java NIO学习笔记 三 散点/收集 和频道转换

    Java NIO散点/收集 Java NIO带有内置的分散/收集支持.散点/收集是读取和写入渠道过程中使用的概念. 从通道散射读取是将数据读入多个缓冲区的读取操作.因此,数据可以从通道“散布”到多个缓 ...

  7. Java NIO学习系列五:I/O模型

    前面总结了很多IO.NIO相关的基础知识点,还总结了IO和NIO之间的区别及各自适用场景,本文会从另一个视角来学习一下IO,即IO模型.什么是IO模型?对于不同人.在不同场景下给出的答案是不同的,所以 ...

  8. java基础学习系列三

    产生随机数 例如 [a,b] Math.random*(b-a+1)+a 公式推算 [3,55]-----[0,52]+3 *53+3

  9. Java NIO 学习笔记(三)----Selector

    目录: Java NIO 学习笔记(一)----概述,Channel/Buffer Java NIO 学习笔记(二)----聚集和分散,通道到通道 Java NIO 学习笔记(三)----Select ...

随机推荐

  1. WPF ListBoxItem模板中添加CheckBox选中问题

    原文:WPF ListBoxItem模板中添加CheckBox选中问题 是这样的,需要一个ListBox来展示照片,并添加一个选中的CheckBox.这就需要对ListBox的ItemTemplate ...

  2. DataGrid数据绑定

    后台数据绑定 用户场景是生成报表,展示公司各员工每个月的绩效 数据结构 包括报表和单个员工绩效两个实体 public class Report { /// <summary> /// 统计 ...

  3. VisualSVN-6.0.1Patch just for VS2017补丁原创发布

    VisualSVN-6.0.1Patch_justforVS2017补丁原创发布 一切尽在发布中.

  4. WPF IDataErrorInfo使用-数据对象上验证

    <Window x:Class="DataBindingExam.MainWindow"        xmlns="http://schemas.microsof ...

  5. 八荣八耻 IT版

    八荣八耻 IT版以可配置为荣,以硬编码为耻:以系统互备为荣,以系统单点为耻:以随时可重启为荣,以不能迁移为耻:以整体交付为荣,以部分交付为耻:以无状态为荣,以有状态为耻:以标准化为荣,以特殊化为耻:以 ...

  6. VS点击调试卡住的问题解决方案(转载)

    本来今天好好的,不知道弄到了什么,调试不了了,一点击立马卡住,就一直在那转,就在网上找了找解决方案,下面给大家列出来几种可能会卡住的问题已经解决方案 1:加载调试符号引起的卡住 解决方案: 在“选项” ...

  7. Oracle序列使用:建立、删除、使用

    Oracle序列使用:建立.删除 在开始讲解Oracle序列使用方法之前,先加一点关于Oracle client sqlplus的使用,就是如果执行多行语句的话一定要加“/”才能表示结束,并执行!本篇 ...

  8. C#二分查找法 破洞百出版本

    二分查找法在数据繁多的数据中查找是一种快速的方法,每次查找最多需要的次数 为2的n次方小于总个数. 当然是有前提的,就是需要把数据先排好序,这里指的都是数值型的数据. 基本思想就是把需要找的值与排序好 ...

  9. myCloudData SDK

    http://www.tmssoftware.com/site/myclouddata.asp http://www.tmssoftware.com/site/myclouddatasdk.asp

  10. Qt 5.6 5.8 vs2015 编译静态库版本(有全部的截图)good

    安装Qt 去Qt官网下载Qt安装包  安装Qt和源码,一定要勾选source选项  添加bin到系统变量  工具 需要python3和 perl. vs2015 第三方工具,到官方下载安装  在命令行 ...