Java NIO框架Netty教程(七)-再谈收发信息次数问题

本文深入探讨了Java NIO的Selector机制及其在Netty框架中的应用,详细解释了消息收发次数不匹配问题的原因,并通过Netty的代码实现进一步解析了消息处理过程。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

  • 在《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. final ChannelBufferFactory bufferFactory =
    20. channel.getConfig().getBufferFactory();
    21. final ChannelBuffer 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

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值