5.6producer如何处理粘包问题

在生产者向服务端发送请求后,服务端的响应可能会出现粘包现象,即多个响应被一起发送。为了解决这个问题,文中介绍了通过在消息头添加消息大小的整数来分隔不同响应的方法。首先读取4字节确定消息体的大小,然后分配相应内存读取完整消息。通过这种方式,可以正确地分离和处理每个独立的响应。
摘要由CSDN通过智能技术生成

生产者向服务端发送请求之后,服务端会向生产者发送一个响应,这时就会遇到粘包和拆包的问题,本节我们来看一下如何处理粘包问题

为什么会有粘包问题

一个请求里面会带有多个响应

多个消息粘在一起发送回来了,正常应该是一条一条发送回来的

我们接着上节的代码看

 //核心代码,处理发送请求的事件
                //selector注册了一个OP_WRITE事件
                //selector还注册了一个OP_READ事件
                if (channel.ready() && key.isWritable()) {
                    //这里就是要往服务端发送数据了
                    Send send = channel.write();
                    if (send != null) {
                        this.completedSends.add(send);
                        this.sensors.recordBytesSent(channel.id(), send.size());
                    }
                }

这里由于selector还注册了一个OP_READ事件,所以下一次循环进来的时候,会走到下面的分支

 /* if channel is ready read from any connections that have readable data */
                if (channel.ready() && key.isReadable() && !hasStagedReceive(channel)) {
                    NetworkReceive networkReceive;
                    //接收服务端发送回来的响应
                    while ((networkReceive = channel.read()) != null)
                        addToStagedReceives(channel, networkReceive);
                }

可以先来看一下如何读取这些响应

    public NetworkReceive read() throws IOException {
        NetworkReceive result = null;

        if (receive == null) {
            receive = new NetworkReceive(maxReceiveSize, id);
        }
        //接收响应
        receive(receive);
        if (receive.complete()) {
            receive.payload().rewind();
            result = receive;
            receive = null;
        }
        return result;
    }
private long receive(NetworkReceive receive) throws IOException {
        //读取数据
        return receive.readFrom(transportLayer);
    }

接下来就要到核心代码了

这里我们先来了解一下解决粘包问题的解决方案

int类型的数(消息大小)消息 int类型的数 消息

主要就是用这样的一个格式来解决

@Deprecated
    public long readFromReadableChannel(ReadableByteChannel channel) throws IOException {
        int read = 0;
        if (size.hasRemaining()) {
            //先读取4字节的数据(代表的就是后面跟着的消息体的大小)
            int bytesRead = channel.read(size);
            if (bytesRead < 0)
                throw new EOFException();
            read += bytesRead;
            if (!size.hasRemaining()) {
                size.rewind();
                int receiveSize = size.getInt();
                if (receiveSize < 0)
                    throw new InvalidReceiveException("Invalid receive (size = " + receiveSize + ")");
                if (maxSize != UNLIMITED && receiveSize > maxSize)
                    throw new InvalidReceiveException("Invalid receive (size = " + receiveSize + " larger than " + maxSize + ")");
               //分配一个内存空间,这个内存空间的大小
                //就是刚刚读出来的那个4字节的int的大小
                this.buffer = ByteBuffer.allocate(receiveSize);
            }
        }
        if (buffer != null) {
            //读取数据
            int bytesRead = channel.read(buffer);
            if (bytesRead < 0)
                throw new EOFException();
            read += bytesRead;
        }

        return read;
    }
 public NetworkReceive read() throws IOException {
        NetworkReceive result = null;

        if (receive == null) {
            receive = new NetworkReceive(maxReceiveSize, id);
        }
        //接收响应
        receive(receive);
        //是否读完一个完整的消息
        if (receive.complete()) {
            receive.payload().rewind();
            result = receive;
            receive = null;
        }
        return result;
    }
 public boolean complete() {
        //size没有剩余空间了 && buffer没有剩余空间了
        return !size.hasRemaining() && !buffer.hasRemaining();
    }
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值