Netty 解决粘包半包问题

Netty 解决粘包/半包问题

解决Tcp粘包问题的主流方法有如下几种

  • 客户端和服务端应用程序在接收数据时按 换行符进行分包 --Netty中的LineBasedFrameDecoder

  • 客户端和服务端应用程序在接收数据时按指定分隔符 进行分包–Netty中的DelimiterBasedFramDecoder

  • 客户端和服务端应用程序在接收数据时按固定长度进行分包–Netty中的FixedLengthFrameDecoder

  • 客户端和服务端应用程序在收发数据时按块编码, (需要自己实现,Netty未提供编码/解码器)

    格式:100 (标志位,第一个字节表示数据的长度)xxx (数据体,100个字节的数据体)

    ​ …

    ​ 0 (标志位,数据结束,数据包中完整的数据已读取完成)

一、按换行符分包

private class ChildChannelHandler extends ChannelInitializer<SocketChannel> {
@Override
protected void initChannel(SocketChannel arg0) throws Exception {
    // 配置按换行符解码器;设置一行的最大长度。
    arg0.pipeline().addLast(new LineBasedFrameDecoder(1024));
    arg0.pipeline().addLast(new TimeServerHandler());
}

二、按特定字符分包

@Override
public void initChannel(SocketChannel ch)
   throws Exception {
    ByteBuf delimiter = Unpooled.copiedBuffer("$_"
       .getBytes());
    // 以$_作为分包
    ch.pipeline().addLast(
       new DelimiterBasedFrameDecoder(1024,
          delimiter));
    ch.pipeline().addLast(new EchoServerHandler());
}

三、固定长度分包

@Override
public void initChannel(SocketChannel ch)
   throws Exception {
    // 每20个字节分一个包
    ch.pipeline().addLast(
       new FixedLengthFrameDecoder(20));
    ch.pipeline().addLast(new EchoServerHandler());
}

小结

具体选用哪种方式由应用的业务场景决定

  1. 如果传输的数据类型是文本,并且文本很小,可以按照换行符/指定分隔符来分包;
  2. 如果传输的是二进制类型(文件上传、下载),可以使用按块编码进行分包;
  3. 固定长度分包,发送端需要对不满长度的包进行填充字符。(不推荐使用)
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值