物联网开发tcp协议之——netty拆包问题

1.前言

tcp协议是物联网开发中比较常见的一种通信协议,而netty则是一tcp通信协议中一个比较优秀的框架。tcp协议是一种长连接的协议,是流式传输的,开发过程中最长遇见的问题就是拆包粘包问题。我目前对接过的物联网系列有智能家居设备,433酒店智控设备,充电桩设备,安防设备,传感器(温度,噪声等等类的),对接设备的过程中有的是和厂家对接,这种需要完全根据厂家的协议走,有的是公司自研的这和设备端嵌入式工程师商议协议即可。对接过程中遇到的一个最大问题就是拆包粘包的过程。下面就目前我遇到的协议中做一个总结:

1.常规协议,netty常规编解码器

使用Netty实现的tcp服务端,由于tcp是流式传输的,故需要选用一个解码器对流式消息进行解码和包分隔,以防收到不正确的包。例如LineBasedFrameDecoder,LengthFieldBasedFrameDecoder,DelimiterBasedFrameDecoder等常用解码器。

比如对接的智能家居设备是公司自研的设备,嵌入式工程师也是我们自己公司的,定义协议的时候,以固定报文结尾,假如固定报文为:0D0ABBBB,那么可使用netty编解码器ByteArrayDecoder,ByteArrayEncoder,附上代码:

public class HwServerInitializer extends ChannelInitializer<SocketChannel> {
//
    @Override
    protected void initChannel(SocketChannel ch) {
        ChannelPipeline pipeline = ch.pipeline();
        ByteBuf delimiter = Unpooled.copiedBuffer(new byte[]{0x0D, 0x0A, (byte)0xBB, (byte)0xBB});
        
        // 以("\n")为结尾分割的 解码器
        pipeline.addLast("framer", new DelimiterBasedFrameDecoder(10000, delimiter));
        // 字符串解码 和 编码
        pipeline.addLast("decoder", new ByteArrayDecoder());
        pipeline.addLast("encoder", new ByteArrayEncoder());
        // 自己的逻辑Handler
        pipeline.addLast("handler", new HwServerHandler());
    }
}

2.不规则协议

有些协议是用常规的编解码器解决不了的,比如下面这一个安防类的协议,

包很长,常规的编解码器解决不了,一直拆包粘包,那么就需要自定义编解码器了,附上自定义解码器的代码

public class MyMessageDecode extends ByteToMessageDecoder {
    private static final Logger log = LoggerFactory.getLogger(MyMessageDecode.class);

    @Override
    protected void decode(ChannelHandlerContext channelHandlerContext, ByteBuf byteBuf, List<Object> list) throws Exception {
        // 标记一下当前的readIndex的位置
        byteBuf.markReaderIndex();
        byte[] magic = new byte[5];
        byteBuf.readBytes(magic);
        byte[] len = new byte[4];
        byteBuf.readBytes(len);
        int lens=Integer.parseInt(MyStringUtil.byteArrayToASCII(len));
        if (byteBuf.readableBytes()<lens-10){
            byteBuf.resetReaderIndex();
            return;
        }
        byte[] data = new byte[lens-9];
        byteBuf.readBytes(data);
        byte[] mesages=new byte[lens];
        System.arraycopy(magic, 0, mesages, 0, 5);
        System.arraycopy(len, 0, mesages, 5, 4);
        System.arraycopy(data, 0, mesages, 9, lens-9);
       // byteBuf.resetReaderIndex();
        list.add(mesages);
    }
}

拆包粘包问题解决

  • 5
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

软件编程工程师

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值