189 8069 5689

TCP粘拆包与Netty代码的示例分析

TCP粘拆包与Netty代码的示例分析,相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。

创新互联专注于企业营销型网站、网站重做改版、四川网站定制设计、自适应品牌网站建设、H5场景定制商城网站开发、集团公司官网建设、成都外贸网站制作、高端网站制作、响应式网页设计等建站业务,价格优惠性价比高,为四川等各大城市提供网站开发制作服务。

TCP是个“流”协议,所谓流,就是没有界限的一串数据。可以想想河里的流水,是连成一片的,其间并没有分界线。TCP底层并不了解上层业务数据的具体含义,它会根据TCP缓冲区的实际情况进行包的划分,所以在业务上认为,一个完整的包可能会被TCP拆分成多个包进行发送,也有可能把多个小的包封装成一个大的数据包发送,这就是所谓的TCP粘包和拆包问题。

TCP粘包或拆包的原因

  1. 应用程序写入的数据大于套接字缓冲区大小,这将会发生拆包。

  2. 应用程序写入数据小于套接字缓冲区大小,网卡将应用多次写入的数据发送到网络上,这将会发生粘包。

  3. 进行MSS(最大报文长度)大小的TCP分段,当TCP报文长度-TCP头部长度>MSS的时候将发生拆包。

  4. 接收方法不及时读取套接字缓冲区数据,这将发生粘包。

拆包和粘包的形式

第一种情况:接收端正常收到两个数据包,即没有发生拆包和粘包的现象,此种情况不在本文的讨论范围内。

TCP粘拆包与Netty代码的示例分析

第二种情况:接收端只收到一个数据包,由于TCP是不会出现丢包的,所以这一个数据包中包含了发送端发送的两个数据包的信息,这种现象即为粘包。这种情况由于接收端不知道这两个数据包的界限,所以对于接收端来说很难处理。

TCP粘拆包与Netty代码的示例分析

第三种情况:这种情况有两种表现形式,如下图。接收端收到了两个数据包,但是这两个数据包要么是不完整的,要么就是多出来一块,这种情况即发生了拆包和粘包。这两种情况如果不加特殊处理,对于接收端同样是不好处理的。

TCP粘拆包与Netty代码的示例分析

TCP粘拆包与Netty代码的示例分析

粘包和拆包的解决办法

  1. 发送端给每个数据包添加包首部,首部中应该至少包含数据包的长度,这样接收端在接收到数据后,通过读取包首部的长度字段,便知道每一个数据包的实际长度了。

  2. 发送端将每个数据包封装为固定长度(不够的可以通过补0填充),这样接收端每次从接收缓冲区中读取固定长度的数据就自然而然的把每个数据包拆分开来。

  3. 可以在数据包之间设置边界,添加特殊符号(如:回车符),这样,接收端通过这个边界就可以将不同的数据包拆分开。

Netty中的代码示例

Netty封装了JDK的NIO,是一个异步事件驱动的网络应用框架,用于快速开发可维护的高性能服务器和客户端。一般开发中并不会用JDK原生NIO,原因如下:

  1. 使用JDK自带的NIO需要了解太多的概念,编程复杂,一不小心bug横飞

  2. Netty底层IO模型随意切换,而这一切只需要做微小的改动,改改参数,Netty可以直接从NIO模型变身为IO模型

  3. Netty自带的拆包解包,异常检测等机制让你从NIO的繁重细节中脱离出来,让你只需要关心业务逻辑

  4. Netty解决了JDK的很多包括空轮询在内的bug

  5. Netty底层对线程,selector做了很多细小的优化,精心设计的reactor线程模型做到非常高效的并发处理

  6. 自带各种协议栈让你处理任何一种通用协议都几乎不用亲自动手

  7. Netty社区活跃,遇到问题随时邮件列表或者issue

  8. Netty已经历各大rpc框架,消息中间件,分布式通信中间件线上的广泛验证,健壮性无比强大

所以,本文选择演示Netty的编解码代码。

在Netty中,我们定义MessageToByteEncoder的继承类,重写其encode函数,来自定义编码器。

public class SocketEncoder extends MessageToByteEncoder {    @Override    protected void encode(ChannelHandlerContext channelHandlerContext, NetPacket msg, ByteBuf byteBuf) throws Exception {        byte body[] = msg.getBody();        int packetLen = body.length;        // 先设置包长度,然后写入二进制数据        byteBuf.writeInt(packetLen);        byteBuf.writeBytes(body);    }}

在Netty中,我们定义ByteToMessageDecoder的继承类,重写其decode函数,用来自定义解码器。

public class SocketDecoder extends ByteToMessageDecoder {    @Override    void decode(ChannelHandlerContext channelHandlerContext, ByteBuf byteBuf, List list) throws Exception {        int bufLen = byteBuf.readableBytes();        // 解决粘包问题(不够一个包头的长度)        // 4字节是报文中使用了一个int表示了报文长度        if (bufLen < 4) {            return;        }        // 标记一下当前的readIndex的位置        byteBuf.markReaderIndex();        int packetLength = byteBuf.readInt();
       // 读到的消息体长度如果小于我们传送过来的消息长度,则resetReaderIndex。重置读索引,继续接收        if (byteBuf.readableBytes() < packetLength) {            // 配合markReaderIndex使用的。把readIndex重置到mark的地方            byteBuf.resetReaderIndex();            return;        }
       NetPacket netPacket = new NetPacket();        netPacket.setPacketLen(packetLength);        // 传送过来数据的长度,满足我们的要求了        byte body[] = new byte[packetLength];        byteBuf.readBytes(body);        netPacket.setBody(body);        list.add(netPacket);    }}

看完上述内容,你们掌握TCP粘拆包与Netty代码的示例分析的方法了吗?如果还想学到更多技能或想了解更多相关内容,欢迎关注创新互联行业资讯频道,感谢各位的阅读!


当前名称:TCP粘拆包与Netty代码的示例分析
新闻来源:http://gzruizhi.cn/article/jcgchi.html

联系我们

您好HELLO!
感谢您来到宜宾网站建设公司,若您有合作意向,请您为我们留言或使用以下方式联系我们, 我们将尽快给你回复,并为您提供真诚的设计服务,谢谢。
  • 电话:028- 86922220 18980695689
  • 商务合作邮箱:631063699@qq.com
  • 合作QQ: 532337155
  • 成都网站设计地址:成都市青羊区锣锅巷31号五金站写字楼6楼

冠赛建站工作室

宜宾冠赛网站建设公司拥有多年以上互联网从业经验的团队,始终保持务实的风格,以"帮助客户成功"为已任,专注于提供对客户有价值的服务。 我们已为众企业及上市公司提供专业的网站建设服务。我们不只是一家网站建设的网络公司;我们对营销、技术、管理都有自己独特见解,冠赛建站采取“创意+综合+营销”一体化的方式为您提供更专业的服务!

冠赛观点

相对传统的宜宾网站建设公司而言,冠赛是互联网中的网站品牌策划,我们精于企业品牌与互联网相结合的整体战略服务。
我们始终认为,网站必须注入企业基因,真正使网站成为企业vi的一部分,让整个网站品牌策划体系变的深入而持久。