Netty-selector

为什么使用Selector

Selector(选择器)是Java NIO中能够检测一到多个NIO通道,并能够知晓通道是否为诸如读写事件做好准备的组件。这样,一个单独的线程可以管理多个channel,从而管理多个网络连接。

Selector创建

Selector selector = Selector.open();

向Selector注册通道

为了将channel和selector配合使用,需要将channel注册到selector上

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

与selector一起使用时,channel必须处于非阻塞状态

register()方法的第二个参数。这是一个“interest集合”,意思是在通过Selector监听Channel时对什么事件感兴趣。

可以监听四种不同类型的事件

1.connect

2.accept

3.read

4.write

通道触发了一个事件意思就是该事件已经就绪

某个channel成功连接到另一个服务器称为-连接就绪

一个server socket channel准备好接收新进入的连接称为-接收就绪

一个有数据可读的通道可以说是-读就绪

等待写数据的通道可以说是-写就绪

如果对不止一种事件感兴趣,可以用位或操作符将常量连接起来

int interestSet = SelectionKey.OP_READ | SelectionKey.OP_WRITE;

SelectionKey

interest集合-感兴趣的集合事件[selectionKey.interestOps()]

ready集合-已经就绪的操作集合[selectionKey.readyOps()]

Channel[selectionKey.channel()]

Selector[selectionKey.selector()]

附加的对象(可选)

通过Selector选择通道

一旦向Selector注册了一或多个通道,就可以调用几个重载的select()方法。这些方法返回你所感兴趣的事件(如连接、接受、读或写)已经准备就绪的那些通道。

select()返回的int值表示有多少通道已经就绪。

select(long timeout)和select()一样,除了最长会阻塞timeout毫秒(参数)。

selectNow()不会阻塞,不管什么通道就绪都立刻返回

WakeUp

某个线程调用select()方法后阻塞了,即使没有通道已经就绪,也有办法让其从select()方法返回。只要让其它线程在第一个线程调用select()方法的那个对象上调用Selector.wakeup()方法即可。阻塞在select()方法上的线程会立马返回。

如果有其它线程调用了wakeup()方法,但当前没有线程阻塞在select()方法上,下个调用select()方法的线程会立即“醒来(wake up)”。

close

用完Selector后调用其close()方法会关闭该Selector,且使注册到该Selector上的所有SelectionKey实例无效。通道本身并不会关闭

示例

selector对注册的channel轮询访问,一单轮询到一个channel有所注册的时间发生,就会报告,交出来一把钥匙,通过钥匙来读取这个channel的内容

//使用selector
Selector selector = Selector.open();
//建立Channel 并绑定到9000端口
ServerSocketChannel channel = ServerSocketChannel.open();
InetSocketAddress address = new InetSocketAddress(InetAddress.getLocalHost(),9000); 
channel.socket().bind(address);
//设定non-blocking方式
channel.configureBlocking(false);
//向selector注册channel以及我们有兴趣的事件
SelectionKey key = channel.register(selector, SelectionKey.OP_READ);
while(true) {
  //selector通过select方法通知我们,我们感兴趣的事件发生了
  int readyChannels = selector.select();
  //如果有我们感兴趣的事件,readyChannels大于0
  if(readyChannels == 0) continue;
  //selector传回一组selectedKeys,我们从这些key的channel方法取得我们注册的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();
  }
}


blog comments powered by Disqus