JAVA NIO客户端主动关闭连接,导致服务器空轮询

当客户端连接关闭时,服务器select()不会阻塞,然后一直分发读就绪操作,且读到的字节长度都是0,这是什么情况。

服务器代码

public class NIOServerTest {

public static void main(String[] args) {

try {

ServerSocketChannel serverChannel = ServerSocketChannel.open();

serverChannel.bind(new InetSocketAddress(666));

serverChannel.configureBlocking(false);

Selector selector = Selector.open();

serverChannel.register(selector,SelectionKey.OP_ACCEPT);

while(true){

int count = selector.select(); //阻塞

if(count>0){

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

Iterator<SelectionKey> iterator = keys.iterator();

while(iterator.hasNext()){

SelectionKey key = iterator.next();

iterator.remove();

if(key.isAcceptable()){

System.out.println("client connect");

ServerSocketChannel serverSocketChannel = (ServerSocketChannel) key.channel();

SocketChannel sc = serverSocketChannel.accept();

sc.configureBlocking(false);

sc.register(selector, SelectionKey.OP_READ);

}

if(key.isReadable()){

SocketChannel socketChannel = (SocketChannel) key.channel();

ByteBuffer buffer = ByteBuffer.allocate(512);

socketChannel.read(buffer);

buffer.flip();

System.out.println("on read size:"+buffer.remaining());

}

}

}

}

} catch (IOException e) {

e.printStackTrace();

}

}

}

客户端代码

public class NIOClientTest{

public static void main(String[] args) throws UnknownHostException, IOException{

try {

Socket socket = new Socket("127.0.0.1",666);

try(OutputStreamWriter output = new OutputStreamWriter(socket.getOutputStream());){

output.write(1);

output.flush();

}catch (Exception e) {

e.printStackTrace();

}

socket.close();

} catch (Exception e1) {

e1.printStackTrace();

}

}

}

回答:

当客户端主动切断连接的时候,服务端 Socket 的读事件(FD_READ)仍然起作用,也就是说,服务端 Socket 的状态仍然是有东西可读,当然此时读出来的字节肯定是 0。

socketChannel.read(buffer) 是有返回值的,这种情况下返回值是 -1,所以如果 read 方法返回的是 -1,就可以关闭和这个客户端的连接了。
SocketChannel.read 的返回值

这种情况也有可能会抛出 IOException,需要捕获异常并判断。

回答:

服务端需要监听客户端的关闭,不管客户端是正常还是异常关闭,服务端都要监听,监听到客户端关闭了,就把这个长链接close就好了。

以上是 JAVA NIO客户端主动关闭连接,导致服务器空轮询 的全部内容, 来源链接: utcz.com/p/181782.html

回到顶部