NIO组件Selector调用实例
*对于nio的非阻塞I/O操作,使用Selector获取哪些I/O准备就绪,注册的SelectionKey集合记录关联的Channel这些信息.SelectionKey记录Channel对buffer的操作方式.
---SelectableChannel,Selector,SelectionKey是nio中Channel操作的3个主要部件.
---对应关系,
一个SelectableChannel,记录了一组注册的SelectionKey[]
一个SelectionKey,关联了一个Channel和一个Selector.
一个Selector,维护着注册的一组SelectionKey
*使用示例
---注,这段代码来自"TCP/IP Sockets in Java",典型示例,我做了注解
package chapter5; import java.io.IOException;
import java.net.InetSocketAddress;
import java.nio.channels.SelectionKey;
import java.nio.channels.Selector;
import java.nio.channels.ServerSocketChannel;
import java.util.Iterator; public class TCPServerSelector {
private static final int BUFSIZE = 256; //Buffer,EchoSelectorProtocol
private static final int TIMEOUT = 3000; //Selector.select(long timeout) public static void main(String[] args) throws IOException {
if(args.length<1){
throw new IllegalArgumentException("Parameter(s): <Port> ...");
} Selector selector =Selector.open(); //创建Selector实例
for(String arg:args){
ServerSocketChannel listnChannel =ServerSocketChannel.open();
listnChannel.socket().bind(new InetSocketAddress(Integer.parseInt(arg)));
listnChannel.configureBlocking(false); //nonblocking
listnChannel.register(selector, SelectionKey.OP_ACCEPT); //Channel注册selector,并告知channel感兴趣的操作
} TCPProtocol protocol =new EchoSelectorProtocol(BUFSIZE);
while(true){ //循环
if(selector.select(TIMEOUT)==0){ //返回准备就绪I/O的SelectionKey数
System.out.println("."); //to do others
continue;
} Iterator<SelectionKey> keyIter =selector.selectedKeys().iterator();//获取已选的SelectionKey集合
while(keyIter.hasNext()){ //遍历key,根据key的类型做相应处理
SelectionKey key =keyIter.next();
if(key.isAcceptable())
protocol.handleAccept(key); if(key.isReadable())
protocol.handleRead(key); //SelectionKey is invalid if it is cancelled, its channel is closed, or its selector is closed.
if(key.isValid() && key.isWritable())
protocol.handleWrite(key); keyIter.remove(); //手动清空,因为Selector只会在已选的SelectionKey集中添加
}
} }
}
package chapter5; import java.io.IOException;
import java.nio.channels.SelectionKey; public interface TCPProtocol {
void handleAccept(SelectionKey key) throws IOException;
void handleRead(SelectionKey key) throws IOException;
void handleWrite(SelectionKey key) throws IOException;
} package chapter5; import java.io.IOException;
import java.nio.ByteBuffer;
import java.nio.channels.SelectionKey;
import java.nio.channels.ServerSocketChannel;
import java.nio.channels.SocketChannel; public class EchoSelectorProtocol implements TCPProtocol { private int bufSize; public EchoSelectorProtocol(int bufSize){
this.bufSize =bufSize;
} @Override
public void handleAccept(SelectionKey key) throws IOException {
SocketChannel clntChan = ((ServerSocketChannel)key.channel()).accept();
clntChan.configureBlocking(false);
//对于已注册Selector的Channel,再次调用就更新注册信息,这里更新了SelectionKey的类型和附件,附件是需要操作的buffer.
clntChan.register(key.selector(), SelectionKey.OP_READ,ByteBuffer.allocate(bufSize));
} @Override
public void handleRead(SelectionKey key) throws IOException {
SocketChannel clntChan =(SocketChannel)key.channel();
ByteBuffer buf =(ByteBuffer) key.attachment(); //获取附件
long bytesRead = clntChan.read(buf);
if(bytesRead==-1) //channel已读到结束位置
clntChan.close();
else if(bytesRead > 0)
key.interestOps(SelectionKey.OP_READ|SelectionKey.OP_WRITE);
} @Override
public void handleWrite(SelectionKey key) throws IOException {
ByteBuffer buf =(ByteBuffer) key.attachment();
buf.flip();
SocketChannel clntChan =(SocketChannel) key.channel();
clntChan.write(buf);
if(!buf.hasRemaining()){
key.interestOps(SelectionKey.OP_READ); //设置Key的兴趣集
}
buf.compact();
} }
看了代码,如何使用就清楚了吧:).
下面再做些细节说明
*Selector
---更新准备好的SelectionKey,移除isValid()为false的SelectionKey
select() //阻塞等待,直至一个channel准备好或调用wakeup()才返回
select(long timeout) //如上,返回条件多了个超时时间
selectNow() //非阻塞,会立刻返回,没有时返回值=0
wakeup() //使得Selector返回
注意,select()会在上次已选择的键集中添加这次的可用键,故在2次select之间,手动移除已处理的SelectionKey.
*SelectionKey
---兴趣操作集,通过它就可以知道channel可以去做哪些事了.有4种类型,如下:
public static final int OP_READ = 1 << 0;
public static final int OP_ACCEPT = 1 << 4;
public static final int OP_WRITE = 1 << 2;
public static final int OP_CONNECT = 1 << 3;
通过SelectionKey.interestOps(int ops)就可以配置这些值
---附件,主要作用是为channel处理提供辅助信息,如上面示例中att为ByteBuffer
SelectionKey.attach(Object ob) //添加附件,另一种方式SelectableChannel.register(Selector sel, int ops, Object att)
SelectionKey.attachment() //获取附件
---SelectionKey.cancel(),永久的注销键,加入Selector的注销集中,在下次select()时被移除
NIO组件Selector调用实例的更多相关文章
- NIO组件Selector工作机制详解(上)
转自:http://blog.csdn.net/haoel/article/details/2224055 一. 前言 自从J2SE 1.4版本以来,JDK发布了全新的I/O类库,简称NIO,其不但 ...
- NIO组件 Selector(选择器)
简介 使用Selector(选择器), 可以使用一个线程处理多个客户端连接. Selector 能够检测多个注册的通道上是否有事件发生(多个Channel以事件的方式可以注册到同一个Selector) ...
- NIO组件Selector详解
Selector(选择器)是Java NIO中能够检测一到多个NIO通道,并能够知晓通道是否为诸如读写事件做好准备的组件.这样,一个单独的线程可以管理多个channel,从而管理多个网络连接. 下面是 ...
- NIO组件Selector工作机制详解(下)
转自:http://blog.csdn.net/haoel/article/details/2224069 五. 迷惑不解 : 为什么要自己消耗资源? 令人不解的是为什么我们的Java的New I/ ...
- NIO的Selector
参考自 Java NIO系列教程(六) Selector Java-NIO-Selector java.nio.channels.Selector NIO新功能Top 10(下) 出发点: 如何管理多 ...
- NIO 概述 与 通信实例
NIO 简述: NIO是在jdk1.4之后加入的一种基于缓冲区(buffer)和通道(channel)的I/O方式, nio是同步非阻塞的i/o模式,同步是指线程不断地轮询i/o事件,非阻塞是在处理i ...
- Java NIO之Selector(选择器)
历史回顾: Java NIO 概览 Java NIO 之 Buffer(缓冲区) Java NIO 之 Channel(通道) 其他高赞文章: 面试中关于Redis的问题看这篇就够了 一文轻松搞懂re ...
- Nio使用Selector客户端与服务器的通信
使用NIO的一个最大优势就是客户端于服务器自己的不再是阻塞式的,也就意味着服务器无需通过为每个客户端的链接而开启一个线程.而是通过一个叫Selector的轮循器来不断的检测那个Channel有消息处理 ...
- WPF中实例化Com组件,调用组件的方法时报System.Windows.Forms.AxHost+InvalidActiveXStateException的异常
WPF中实例化Com组件,调用组件的方法时报System.Windows.Forms.AxHost+InvalidActiveXStateException的异常 在wpf中封装Com组件时,调用组件 ...
随机推荐
- 在Laravel5.* 中使用 AdminLTE
在Laravel5.* 中使用 AdminLTE AdminLTE是一个很棒的单纯的由 HTML 和 CSS 构建的后台模板,在这片文章中,我将讲述如何将 AdminLTE 和 Laravel 优雅的 ...
- C# 翻页设计:首页,上一页,下一页,末页 ,跳转
int pageSize = 0; //每页显示行数 int nMax = 0; //总记录数 int pageCount = 0; //页数=总记录数/每页显示行数 int pageCurrent ...
- 问自己----也是自己该怎么走的路(phper)
1.首先看了PHP的源码API函数,对于许多口水仗的争论一笑而过,只是停留在脚本级别上的什么效率,安全...之争完全就是无稽之谈,没有深入理解API,所有的争论都是臆测和不科学的态度.你做了吗? 2. ...
- VS2012 独占编辑 设置
用VS2008建立的项目就有(禁止其他用户签出和签入) 为什么VS2012的项目就没有了呢??(管理员说他什么都没设置)VS2012: 两个项目同样是用VS2012打开的,而第一个是用VS2008创建 ...
- Js Carousel
http://getbootstrap.com/javascript/#carousel http://owlgraphic.com/owlcarousel/#demo https://www.mob ...
- JDK神坑:JAVA中Calendar的月份Month少1
很多朋友在使初次使用Calendar时,会发现月份莫名其妙对不上,显示的结果总是比预期中小1个月,检查好几遍也没发现程序有错,于是开始抓狂.其实这个时候,只要去看JDK就会明白问题所在.JDK告诉我们 ...
- SQL Server强制删除复制发布
原文地址:http://blog.csdn.net/leamonjxl/article/details/7352208 SQL Server 中 存在以前(系统还原前)的发布内容,使用鼠标->右 ...
- python学习之---匿名函数,返回函数,偏函数
1. 返回函数: 所谓的返回函数,指的是函数作为返回值.高阶函数除了可以接受函数作为参数外,同样可以接受函数作为结果返回.以下是一个可变参数的求和例子,一般求和函数是如此这般定义的: >> ...
- Stanford Parser学习入门(2)-命令行运行
在Stanford parser目录中已经定义了一部分命令行工具以及图形界面,本文将介绍如何在windows使用这些工具进行语法分析,Linux下也有shell可以使用. 关于如何搭建环境请参考上一篇 ...
- sql server中创建链接服务器图解教程
转自sql server中创建链接服务器图解教程 1.展开服务器对象-->链接服务器-->右击"新建链接服务器" 注意:必须以数据库管理员身份登录(通常也就是sa帐号) ...