Netty-ProtobufVarint32

效果

  • ProtobufVarint32LengthFieldPrepender编码器用于在数据最前面添加Varint32,表示数据长度

    Untitled

  • ProtobufVarint32FrameDecoder是相对应的解码器

    Untitled

Varint32

讲编码器之前,先来讲讲什么是VarInt32(vary int 32),即:可变长的int

在java里,int的长度固定为 4 byte,即 32 bits,最高位为符号位。

而Varint32则不固定长度,最小 1 byte,最大 5 byte,每个byte的最高位如果为1表示下一个byte依然属于Varint32的,为0表示Varint32到当前byte结束。

所以在Varint32中,每个byte只有7bit存储数据。

下面以 398 举例:

// 数字398的二进制表示
110001110
// 在java中int的二进制表示
00000000 00000000 00000001 10001110
// Varint32的二进制表示
10001110 00000011

转换步骤如下:

  1. 398的二进制表示为110001110,总共有9位
  2. 因为398长度为9bit,大于7,所以在Varint32中需要2个byte来存储
  3. Varint32第一个byte存398的低7位0001110,最高位置1表示还未存储完成,即:10001110
  4. Varint32第二个byte存398的后面两位11,最高位置0表示已存储完成,00000011

最后,别问我负数怎么表示,问神奇的海螺。

ProtobufVarint32LengthFieldPrepender

此编码器的作用,就是将数据长度从int转成Varint32,并添加在数据流的最前面。

  1. 查看源码,其入口为decode方法:

    @Override
    protected void encode(
            ChannelHandlerContext ctx, ByteBuf msg, ByteBuf out) throws Exception {
    		// 获取数据长度
        int bodyLen = msg.readableBytes();
    		// 计算int转成Varint32所需字节
        int headerLen = computeRawVarint32Size(bodyLen);
    		// 安全扩充缓冲区
        out.ensureWritable(headerLen + bodyLen);
    		// 将bodyLen转成Varint32并写入
        writeRawVarint32(out, bodyLen);
    		// 写入原有的数据msg
        out.writeBytes(msg, msg.readerIndex(), bodyLen);
    }
    
  2. 先来看看如何计算int转成Varint32所需字节

    static int computeRawVarint32Size(final int value) {
    		// value的低7位有数据,其余位为0
        if ((value & (0xffffffff <<  7)) == 0) {
            return 1;
        }
    		// value的低14位有数据,其余位为0
        if ((value & (0xffffffff << 14)) == 0) {
            return 2;
        }
        if ((value & (0xffffffff << 21)) == 0) {
            return 3;
        }
        if ((value & (0xffffffff << 28)) == 0) {
            return 4;
        }
        return 5;
    }
    
    // 0xffffffff 的二进制表示
    11111111 11111111 11111111 11111111
    // 0xffffffff << 7 的二进制表示
    11111111 11111111 11111111 10000000
    // 假设value为100,二进制表示
    00000000 00000000 00000000 01100100
    // value与0xffffffff << 7按位与,即value & (0xffffffff <<  7)
    11111111 11111111 11111111 10000000
    &
    00000000 00000000 00000000 01100100
    =
    00000000 00000000 00000000 00000000 // 结果等于0
    // 假设value为398,value与0xffffffff << 7按位与
    11111111 11111111 11111111 10000000
    &
    00000000 00000000 00000001 10001110
    =
    00000000 00000000 00000001 10000000 // 结果等于384,大于0
    
  3. 再来看看将bodyLen转成Varint32并写入

    static void writeRawVarint32(ByteBuf out, int value) {
        while (true) {
    				// value的低7位有数据,其余位为0
            if ((value & ~0x7F) == 0) {
                out.writeByte(value);
                return;
            } else {
    						// 取value的低7位,最高位置1
                out.writeByte((value & 0x7F) | 0x80);
    						// 右移7位
                value >>>= 7;
            }
        }
    }
    

ProtobufVarint32FrameDecoder

此解码器的作用,是将Varint32 + data,转换成 data

  1. 源码入口

    @Override
    protected void decode(ChannelHandlerContext ctx, ByteBuf in, List<Object> out)
            throws Exception {
    		// 标记读取索引,用户后续的恢复
        in.markReaderIndex();
    		// 读取前的索引位置
        int preIndex = in.readerIndex();
    		// 读取字节,将Varint32转成int
        int length = readRawVarint32(in);
    		// 读取后索引位置等于读取前,表示读取不成功
        if (preIndex == in.readerIndex()) {
            return;
        }
        if (length < 0) {
            throw new CorruptedFrameException("negative length: " + length);
        }
    		// 如果netty读取到的字节长度不满足数据长度,则重置读取索引
        if (in.readableBytes() < length) {
            in.resetReaderIndex();
        } else {
            out.add(in.readRetainedSlice(length));
        }
    }
    
  2. 读取字节,将Varint32转成int

    private static int readRawVarint32(ByteBuf buffer) {
        if (!buffer.isReadable()) {
            return 0;
        }
        buffer.markReaderIndex();
        byte tmp = buffer.readByte();
    		// tmp >= 0,则byte最高位为0,证明Varint32长度为1byte
        if (tmp >= 0) {
            return tmp;
        } else {
    				// result取temp的低7位
            int result = tmp & 127;
    				// Varint还没结束,但netty读不到更多字节了,则返回
            if (!buffer.isReadable()) {
                buffer.resetReaderIndex();
                return 0;
            }
    				// 读取下一个字节,并判断Varint是否在该字节结束
            if ((tmp = buffer.readByte()) >= 0) {
                result |= tmp << 7;
            } else {
    						// 如果Varint在第二个字节还没结束,则取第二个字节的低7位
                result |= (tmp & 127) << 7;
                if (!buffer.isReadable()) {
                    buffer.resetReaderIndex();
                    return 0;
                }
                if ((tmp = buffer.readByte()) >= 0) {
                    result |= tmp << 14;
                } else {
                    result |= (tmp & 127) << 14;
                    if (!buffer.isReadable()) {
                        buffer.resetReaderIndex();
                        return 0;
                    }
                    if ((tmp = buffer.readByte()) >= 0) {
                        result |= tmp << 21;
                    } else {
                        result |= (tmp & 127) << 21;
                        if (!buffer.isReadable()) {
                            buffer.resetReaderIndex();
                            return 0;
                        }
                        result |= (tmp = buffer.readByte()) << 28;
                        if (tmp < 0) {
                            throw new CorruptedFrameException("malformed varint.");
                        }
                    }
                }
            }
            return result;
        }
    }
    
Netty-socketio是一个用于构建实时通信应用程序的开源框架。它基于Netty框架,通过使用WebSocket协议来实现异步、高性能的网络通信。Netty-socketio具有以下特点: 1. 强大的异步处理能力:Netty-socketio采用事件驱动的方式处理客户端请求,能够高效地处理大量并发连接。 2. 完善的消息传递机制:Netty-socketio提供了灵活的消息传递方式,支持广播、点对点、房间等不同的消息发送方式,满足不同场景下的通信需求。 3. 多协议支持:Netty-socketio不仅支持WebSocket协议,还支持其他常用的协议,如TCP、HTTP等,便于与现有的系统集成。 4. 可扩展性强:Netty-socketio提供了丰富的拓展接口,用户可以根据自己的需求定制和扩展框架的功能。 5. 易于使用:Netty-socketio提供了简洁的API和丰富的文档,可以快速上手使用,并提供了相应的示例代码,方便开发者学习和理解。 对于客服应用来说,Netty-socketio作为一个实时通信框架,可以用于构建在线客服聊天系统。通过使用Netty-socketio,我们可以实现客户与客服人员之间的实时消息传递,支持文字、图片、文件等多种类型的消息。客户可以通过网页或移动端应用与客服人员进行沟通,实时解决问题,提升用户体验。 Netty-socketio提供了强大的异步处理能力和全双工通信机制,能够处理大量并发连接,并保持连接的稳定性和可靠性。同时,它的多协议支持和可扩展性强的特点,使得我们可以根据自己的业务需求进行定制和拓展,满足不同客服场景下的通信需求。 总之,Netty-socketio作为一个强大的实时通信框架,为客服应用提供了一种高效、稳定的解决方案,帮助企业构建更好的客服系统,并提升客户的满意度。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值