-
在《Java NIO框架Netty教程(五)- 消息收发次数不匹配的问题》里我们试图分析一个消息收发次数不匹配的问题。当时笔者还是心存疑惑的。所以决定先学习一下Java NIO的Selector机制。
经过简单的了解,笔者大胆的猜测和“武断”一下该问题的原因。
首先,Selector机制让我们注册一个感兴趣的时间,然后只要有该时间发生,就会传递给接收端。我们写了三次,接收端一定会出发三次的。
然后,Netty实现机制里,有个Buffer缓冲池,把收到的信息都缓存在里面,通过一个线程统一处理。也就是我们看到的那个buffer的处理过程。
Netty的设置中,有一个一次性最多读取字节大小的设定。并且,事件的触发是在处理过缓冲池中的消息之后。我们再来回顾一下Netty中读取信息的那段代码:
01.ByteBuffer bb = recvBufferPool.acquire(predictedRecvBufSize);02.try{03.while((ret = ch.read(bb)) >0) {04.readBytes += ret;05.if(!bb.hasRemaining()) {06.break;07.}08.}09.failure =false;10.}catch(ClosedChannelException e) {11.// Can happen, and does not need a user attention.12.}catch(Throwable t) {13.fireExceptionCaught(channel, t);14.}15.16.if(readBytes >0) {17.bb.flip();18.19.finalChannelBufferFactory bufferFactory =20.channel.getConfig().getBufferFactory();21.finalChannelBuffer buffer = bufferFactory.getBuffer(readBytes);22.buffer.setBytes(0, bb);23.buffer.writerIndex(readBytes);24.25.recvBufferPool.release(bb);26.27.// Update the predictor.28.predictor.previousReceiveBufferSize(readBytes);29.30.// Fire the event.31.fireMessageReceived(channel, buffer);32.}else{33.recvBufferPool.release(bb);34.}
可以看到,如果没有读取到字节是不会触发事件的,所以我们可能会收到2次或者3次信息。(如果发的快,解析的慢,后两次信息,一次性读取了,就2次,如果发送间隔长,分次解析,就收到3次。)原因应该就是如此。跟我们开始猜的差不多,只是不敢确认。
阅读下一篇:Java NIO框架Netty教程(八) Object对象传递 http://www.it165.net/pro/html/201207/3302.html
Java NIO框架Netty教程(七)-再谈收发信息次数问题
最新推荐文章于 2024-04-28 09:12:30 发布
本文深入探讨了Java NIO的Selector机制及其在Netty框架中的应用,详细解释了消息收发次数不匹配问题的原因,并通过Netty的代码实现进一步解析了消息处理过程。
435

被折叠的 条评论
为什么被折叠?



