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还有其他一些解决方法;其中一个是选择(超时)但我试图找出错误的位置.

代码

第一线:

  1. static Selector selector = Selector.open();
  2. while(true) {
  3. int n = selectorGlobal.select();
  4. selectorKeySet = selectorGlobal.selectedKeys().iterator();
  5. while (selectorKeySet.hasNext()) {
  6. selectionKey = selectorKeySet.next();
  7. if (selectionKey.isReadable()) {
  8. //do something
  9. }
  10. if(selectionKey.isAcceptable()) {
  11. //accept
  12. }
  13. }
  14. }

第二线:

  1. while (itr.hasNext()) {
  2. data = (String) itr.next();
  3. String IP = data.get(0);
  4. String Port = data.get(1);
  5. SocketChannel socketChannel = SocketChannel.open();
  6. socketChannel.configureBlocking(true);
  7. boolean isConnected = socketChannel.connect(new InetSocketAddress(IP,Port));
  8. ClassName.selector.wakeup();
  9. SelectionKey selectionKey = SelectSockets.registerChannel(ClassName.selector,socketChannel,SelectionKey.OP_READ);
  10. }
最佳答案
如果您在选择器中注册它,则可能不希望线程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.

猜你在找的Java相关文章