java nio Selector唤醒

前端之家收集整理的这篇文章主要介绍了java nio Selector唤醒前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。

请指出/给我一个selector.wakeup()的工作示例;两个线程之间的方法.

我试图创建一个简单的程序,其中一个线程正在等待selector.select()方法.第二个线程创建一些套接字并尝试向选择器注册;第一个线程被阻止的位置.

因此我需要使用选择器的唤醒方法,但不知何故第一个线程不会出现阻塞模式.

唤醒方法javadoc说明:

If another thread is currently blocked
in an invocation of the
Selector.select() or
Selector.select(long) methods then
that invocation will return
immediately.

P.S还有其他一些解决方法;其中一个是选择(超时)但我试图找出错误的位置.

代码

第一线:

static Selector selector = Selector.open();
while(true) {
   int n = selectorGlobal.select();
   selectorKeySet = selectorGlobal.selectedKeys().iterator();
   while (selectorKeySet.hasNext()) {
      selectionKey = selectorKeySet.next();
      if (selectionKey.isReadable()) {
         //do something
      }
      if(selectionKey.isAcceptable()) {
         //accept
      }
   }
}

第二线:

while (itr.hasNext()) { 
   data = (String) itr.next();
   String IP = data.get(0);
   String Port = data.get(1);

   SocketChannel socketChannel = SocketChannel.open();
   socketChannel.configureBlocking(true);
   boolean isConnected = socketChannel.connect(new InetSocketAddress(IP,Port));
   ClassName.selector.wakeup();
   SelectionKey selectionKey = SelectSockets.registerChannel(ClassName.selector,socketChannel,SelectionKey.OP_READ);

}
最佳答案
如果您在选择器中注册它,则可能不希望线程2中的套接字被阻塞(因为选择器用于非​​阻塞I / O).我认为通常的做法是让选择器处理与OP_CONNECT的连接(使用SocketChannel.finishConnection()).

看起来你可能在这里有潜在的竞争条件.想象一下这一系列事件:

>线程1:selector.select()
> …时间过去了……
>线程2:Thread1.selector.wakeup()
>线程1:检查密钥的可接受性
>线程1:检查键是否可读
>线程1:循环
>线程1:selector.select()
>线程2:尝试在选择器中注册(但这对于select()来说太迟了)

我建议让线程2设置一个SocketChannel,将它存放在线程1可以获取它的地方(确保在执行此操作时是线程安全的),然后唤醒选择器,让它检查它在线程1中的现有键并让线程1在再次调用Selector.select()之前注册新的SocketChannel.

原文链接:https://www.f2er.com/java/437600.html

猜你在找的Java相关文章