Netty处理半包粘包的解码思路

Netty处理半包粘包的解码思路

前言

写netty通讯的都不免遇到半包粘包情况。
那么半包粘包是什么意思呢?
我们可以把客户端与服务端的网络通讯看作两个人聊天


我们可以把客户端与服务端的网络通讯看作两个人聊天
首先对于半包的情况

假设A要告诉B 我喜欢唱跳RAP篮球
由于某些不明原因,A无法一次性说完这句话,只说了 我喜欢唱跳,后面 RAP篮球 咽了一口口水才说出来
上述这种情况就是半包,终端一次接收到的包不是完整的包,只有一部分,需要等待接收完毕


另一种情况粘包
A对B说 我喜欢唱跳RAP篮球music
明显这里 唱跳RAP篮球的时候这句话就结束了,但是后面还有一句 music
等于是两句话合到一块说了,我们需要断句分开处理
上述情况就是粘包,终端一次收到的包是两包或者更多包,需要分割解析

在物联网通讯中,硬件网络原因或者配置原因导致每次发送的报文不固定,半包粘包有可能同时存在是非常常见的。
正确解析一包完整的报文对完成通讯任务不可或缺


本次demo代码传送门:netty-decode
直接上代码 看注释就完事

import io.netty.buffer.ByteBuf;
import io.netty.buffer.ByteBufUtil;
import io.netty.channel.ChannelHandlerContext;
import io.netty.handler.codec.ByteToMessageDecoder;
import lombok.extern.slf4j.Slf4j;

import java.util.List;

@Slf4j
public class DemoDecoder extends ByteToMessageDecoder {

    // 最大包长
    public static final int MAX_BUFFER = 4096;
    // 包开始标识 55
    public static final byte HEAD = 0x55;
    // 包结束标识 45
    public static final byte TAIL = 0x45;

    @Override
    protected void decode(ChannelHandlerContext ctx, ByteBuf in, List<Object> out) {
        /*打印16进制数据*/
        log.info("hex data:[{}] , channel:[{}]", ByteBufUtil.hexDump(in).toUpperCase(), ctx.channel().id());

        if (in.readableBytes() > MAX_BUFFER) {
            // 当收到的包超过最大包长,认为是异常数据 直接丢弃
            in.skipBytes(in.readableBytes());
            // mark一下可读的位置 避免重置可读下标时重置到了认为要丢弃的部分
            in.markReaderIndex();
            return;
        }

        if (in.readByte() != HEAD) {
            // 如果不是开始符的包不接收
            in.skipBytes(in.readableBytes());
            in.markReaderIndex();
            return;
        }

        /*
         *   这里应该加私有协议的判断,走业务判断是不是一个正常的协议包,如果不是依旧像上面一样丢弃
         *   由于是demo就简单认为包是定长的 包括包头包尾一共 45个字节
         */
        if (in.readableBytes() < 44) {
            // 当前接收到的包大小不足一个完整的包 原因有很多网络问题网卡优化等,导致服务端收到的包是一个半包,并不完整
            // 此时需要返回等待接收剩余的包,不然业务层不能解析一个不完整的包
            in.resetReaderIndex();
            /*
             *   由于没有在out add新对象
             *   所以不会进入decode循环
             *   下一次这个通道继续发数据时,会在当前in的基础上继续写入 直接读取in即可
             */
            return;
        }

        if (in.readableBytes() >= 44) {
            // 当前接收到的包已经达到一个完整包的长度
            in.skipBytes(43);// 读取正文部分
            if (in.readByte() != TAIL) {
                // 如果包尾不是45结束的,认为这不是一个正确的包 直接丢弃
                in.skipBytes(in.readableBytes());
                in.markReaderIndex();
                return;
            } else {
                // 简单的包头包尾校验完成后 可以把这一段数据转对象或者直接把bytebuf传给下一个处理器
                in.resetReaderIndex(); //校验的时候读过了这个流需要重置才能重新读

                ByteBuf byteBuf = in.readBytes(45); // 因为设定是45包长 所以直接读45
                in.markReaderIndex();// 已经读到完整包了 mark一下防止下次reset又把读完的包再读一次

                /*可能后面还有 out add过后会进入下一个循环继续读后面的*/
                out.add(byteBuf);
                return;

            }
        }


        /*通过 out.add 可以把这里处理好的数据传到下一个handler*//*
        out.add(in);

        *//*
         * 添加到out之后,in必须至少读1字节 防止由decoder引起的无限循环的机制
         * 这里全部读取当前流中所有字节
         * *//*
        in.skipBytes(in.readableBytes());*/
    }
}

测试

设定完整16进制报文为
55005B0114543A26030B2C55001D68001B0000001D680000000F01190250150A8A0000160B17023936D516E545
55开头 45结尾

模拟发包
测试1:分三段

channelFuture.channel().write(ByteUtils.getSendBuffer("55005B0114543A26030B2C55001D68"));
channelFuture.channel().write(ByteUtils.getSendBuffer("001B0000001D680000000F01190250150A8A0000160B17023936D516E5"));
channelFuture.channel().write(ByteUtils.getSendBuffer("45"));

测试2:既有半包也有粘包

channelFuture.channel().write(ByteUtils.getSendBuffer("55005B0114543A26030B2C55001D68"));
channelFuture.channel().write(ByteUtils.getSendBuffer("001B0000001D680000000F01190250150A8A0000160B17023936D516E5"));
channelFuture.channel().write(ByteUtils.getSendBuffer("4555005B0114543A26030B2C55001D68001B0000001D680000000F01190250150A8A0000160B17023936D516E5"));
channelFuture.channel().writeAndFlush(ByteUtils.getSendBuffer("45"));

实测发送了3包结果应该是客户端打印3个ok

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,569评论 4 363
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,499评论 1 294
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 109,271评论 0 244
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,087评论 0 209
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,474评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,670评论 1 222
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,911评论 2 313
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,636评论 0 202
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,397评论 1 246
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,607评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,093评论 1 261
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,418评论 2 254
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,074评论 3 237
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,092评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,865评论 0 196
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,726评论 2 276
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,627评论 2 270

推荐阅读更多精彩内容