Netty进阶-协议设计与解析

Netty进阶

    • 一、黏包半包的深入理解(本质原因:TCP是流式协议,消息无边界)
      • 1、TCP滑动窗口
      • 2、协议设计与解析
        • 2.1、Redis协议
        • 2.2、HTTP协议
        • 2.3、自定义协议
          • 2.3.1、自定义协议要求
          • 2.3.2、自定义消息对象(编解码器、消息抽象类、具体消息类)
          • 2.3.3、TestMessageCodec:测试案例
          • 2.3.4、半包问题出现及解决
        • 2.4、@Sharable:可共享handler

一、黏包半包的深入理解(本质原因:TCP是流式协议,消息无边界)

![在这里插入图片描述](https://img-blog.csdnimg.cn/direct/533b7a52da0c4bb6906198d2276086e3.png
在这里插入图片描述

1、TCP滑动窗口

在这里插入图片描述
tcp层面分析黏包半包现象的原因:
1)半包
(1)(tcp层)当接收方的窗口用到一半时用完了,这时候去读就拿到的是半包数据
(2)(链路层)mss限制:当发送的数据超过Mss限制后,会将数据切分发送,就会造成半包
(3)(网络层)mtu 155-40
2)黏包(tcp层)
(1)当接收方的窗口范围较大有空闲时,就会过多的接收数据从而导致黏包
(2)Nagle算法:尽可能多的去发送数据,不要因为发送数据过少导致效率过低可能会造成黏包

2、协议设计与解析

2.1、Redis协议

(1)协议说明

redis对于整个命令会看成一个数组。

例:set key value

//举例:set name zhangsan   //下面每个命令都由一个回车符、换行符分割 字节对应13,10
*3
$3
set
$4
name 
$8
zhangsan
*3:首先需要让你发送数组的长度 *表示的是命令的数量,3则是命令组成的长度。
$3:$表示的是某个命令参数的长度,3表示该命令参数长度为3。
每个命令参数都由\r\n来进行分割

(2)案例

 使用redis协议模拟与redis服务端进行通信,执行一条set、get命令
package cn.com.agree.netty.demo.sthalfpackage.protocolAnalysis;import io.netty.bootstrap.Bootstrap;
import io.netty.buffer.ByteBuf;
import io.netty.channel.Channel;
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.ChannelInboundHandlerAdapter;
import io.netty.channel.ChannelInitializer;
import io.netty.channel.nio.NioEventLoopGroup;
import io.netty.channel.socket.nio.NioSocketChannel;
import lombok.extern.slf4j.Slf4j;import java.nio.charset.Charset;/*** @version 1.0* @ClassName TestRedis* @Description TODO 类描述* @date 2024/4/30  3:58 下午**/
@Slf4j
public class TestRedis {public static void main(String[] args) throws InterruptedException {byte[] LINE = {13, 10};//两个字节表示回车,换行NioEventLoopGroup group = new NioEventLoopGroup();Channel channel = new Bootstrap().group(group).channel(NioSocketChannel.class).handler(new ChannelInitializer<NioSocketChannel>() {@Overrideprotected void initChannel(NioSocketChannel ch) throws Exception {ch.pipeline().addLast(new ChannelInboundHandlerAdapter() {@Overridepublic void channelActive(ChannelHandlerContext ctx){//向redis服务端发送一个写指令:set name changluset(ctx);//向redis服务端发送一个读指令:get nameget(ctx);}@Overridepublic void channelRead(ChannelHandlerContext ctx, Object msg) throws Exception {ByteBuf buf = (ByteBuf) msg;log.debug("收到 {} , 消息为:{}", ctx.channel(), buf.toString(Charset.defaultCharset()));super.channelRead(ctx, msg);}//执行set命令:set name changluprivate void set(ChannelHandlerContext ctx){final ByteBuf buffer = ctx.alloc().buffer();buffer.writeBytes("*3".getBytes());buffer.writeBytes(LINE);buffer.writeBytes("$3".getBytes());buffer.writeBytes(LINE);buffer.writeBytes("set".getBytes());buffer.writeBytes(LINE);buffer.writeBytes("$4".getBytes());buffer.writeBytes(LINE);buffer.writeBytes("name".getBytes());buffer.writeBytes(LINE);buffer.writeBytes("$8".getBytes());buffer.writeBytes(LINE);buffer.writeBytes("zhangsan".getBytes());buffer.writeBytes(LINE);ctx.writeAndFlush(buffer);}//执行get命令:get nameprivate void get(ChannelHandlerContext ctx){final ByteBuf buffer = ctx.alloc().buffer();buffer.writeBytes("*2".getBytes());buffer.writeBytes(LINE);buffer.writeBytes("$3".getBytes());buffer.writeBytes(LINE);buffer.writeBytes("get".getBytes());buffer.writeBytes(LINE);buffer.writeBytes("$4".getBytes());buffer.writeBytes(LINE);buffer.writeBytes("name".getBytes());buffer.writeBytes(LINE);ctx.writeAndFlush(buffer);}});}}).connect("127.0.0.1", 6379).sync().channel();log.debug("客户端连接成功:{}", channel);channel.closeFuture().addListener(future -> {group.shutdownGracefully();});}
}

结果展示:
在这里插入图片描述
在这里插入图片描述

2.2、HTTP协议

2.2.1、认识HttpServerCodec

HttpServerCodec:是一个编解码处理器,处理入站、出站处理器。入站处理器会对http请求进行解码

//CombinedChannelDuplexHandler组合其他两个handler,分别是InBound和OutBound 编解码处理器
public final class HttpServerCodec extends CombinedChannelDuplexHandler<HttpRequestDecoder, HttpResponseEncoder>implements HttpServerUpgradeHandler.SourceCodec {

使用方式:

ch.pipeline().addLast(new HttpServerCodec());

浏览器发送一次请求(无论什么方法请求)实际上会解析成两部分:

若是我们重写channelRead方法,那么一个http请求就会走两次该handler方法,每次执行方法其中的Object
msg分别为不同部分的解析对象

DefaultHttpRequest:解析出来请求行和请求头。
LastHttpContent$1:表示请求体。(即便是get请求,请求体内容为空也会专门解析一个请求体对象)

情况一:若是想区分请求头、请求行走的handler那么就需要写一个简单的判断:

ch.pipeline().addLast(new ChannelInboundHandlerAdapter() {@Overridepublic void channelRead(ChannelHandlerContext ctx, Object msg) throws Exception {//DefaultHttpRequest实现了HttpRequest接口if (msg instanceof HttpRequest){System.out.println("请求行、头");//LastHttpContent实现了HttpContent接口}else if (msg instanceof HttpContent){System.out.println("请求体");}super.channelRead(ctx, msg);}
});

在这里插入图片描述
在这里插入图片描述
情况二:若是我们只对某个特定类型感兴趣的话,例如只对解析出来的DefaultHttpRequest请求体对象感兴趣,可以实现一个SimpleChannelInboundHandler

//②若是只对HTTP请求的请求头感兴趣,那么实现SimpleChannelInboundHandler实例,指明感兴趣的请求对象为HttpRequest(实际就是DefaultHttpRequest)
ch.pipeline().addLast(new SimpleChannelInboundHandler<HttpRequest>() {@Overrideprotected void channelRead0(ChannelHandlerContext ctx, HttpRequest msg) throws Exception {log.debug("解析对象类型:{}", msg.getClass());log.debug(msg.uri());//进行响应返回//①构建响应对象final DefaultFullHttpResponse response =new DefaultFullHttpResponse(msg.protocolVersion(), HttpResponseStatus.OK);// 响应内容final byte[] content = "<h1>Hello,world!</h1>".getBytes();//设置响应头:content-length:内容长度。不设置的话浏览器就不能够知道确切的响应内容大小则会造成一直没有处理完的现象response.headers().setInt(HttpHeaderNames.CONTENT_LENGTH, content.length);response.content().writeBytes(content);//②写会响应ctx.writeAndFlush(response);}
});

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

  • 应该是由于没有指定Content-length,浏览器不能够保证当前内容是全部内容,就会出现一直转圈圈等待的效果,并且响应结果也不会出现。

  • 若是设置了的话就不会出现转圈圈现象以及检查response也会有内容:
    在这里插入图片描述

2.3、自定义协议
2.3.1、自定义协议要求

下面是自定义协议案例的协议内容:

魔数(四个字节):用来在第一时间判定是否是无效数据包
版本号(一个字节):可以支持协议的升级
序列化算法(一个字节):消息正文到底采用哪种序列化反序列化方式,可以由此扩展,例如:json、protobuf、hessian、jdk
指令类型(一个字节):是登录、注册、单聊、群聊... 跟业务相关
请求序号(四个字节):为了双工通信,提供异步能力
对齐填充(一个字节):除正文长度和消息正文凑满16个字节,也就是2的n次方
正文长度(四个字节):正文的长度
消息正文:正文内容(根据序列化算法进行序列化成字节)
  • 魔数:一般发送的头几个都是数字都是魔数,例如Java的二进制字节码的起始8个字节就是魔数。

  • 版本号:一般用于支持协议的升级,若是协议升级可能其中的字段或者其他内容有所更改。

  • 序列化算法:一般指的是消息正文,对消息正文采用特殊的格式。JDK的话缺点就是不能跨平台;谷歌出品的protobuf,其与hession都是二进制的,其可读性不好,但是字节数占用最少性能更高。

  • 指令类型:消息是个什么类型,这与业务相关。 请求序号:例如发送是1 2 3,但是由于网络或其他因素受到的时候不是1 2 3顺序。

  • 正文长度:通过正文长度知道正文接下来需要读取多少字节

2.3.2、自定义消息对象(编解码器、消息抽象类、具体消息类)

在这里插入图片描述

  • Message:消息抽象类,定义了消息相关的一些字段内容。
  • LoginRequestMessage:一条业务消息,实现了Message抽象类,是登陆请求消息的抽象。
  • MessageCodec:实现了ByteToMessageCodec执行器,需要传入一个泛型,该泛型就是你要将Bytebuf转换的对象,并且其中需要你重写编解码方法,也就是解析、封装你自定义的一些协议

Message:

package cn.com.agree.netty.demo.sthalfpackage.protocolAnalysis.custom;import lombok.Data;import java.io.Serializable;
import java.util.HashMap;
import java.util.Map;/*** @version 1.0* @ClassName Message* @Description TODO 类描述* @date 2024/4/30  5:56 下午**/
@Data
public  abstract class Message implements Serializable {/*** 根据消息类型字节,获得对应的消息 class* @param messageType 消息类型字节* @return 消息 class*/public static Class<? extends Message> getMessageClass(int messageType) {return messageClasses.get(messageType);}private int sequenceId;private int messageType;public abstract int getMessageType();public static final int LoginRequestMessage = 0;private static final Map<Integer, Class<? extends Message>> messageClasses = new HashMap<>();}

MessageCodec:实现了对自定义协议的编解码

package cn.com.agree.netty.demo.sthalfpackage.protocolAnalysis.custom;import io.netty.buffer.ByteBuf;
import io.netty.channel.ChannelHandlerContext;
import io.netty.handler.codec.ByteToMessageCodec;
import lombok.extern.slf4j.Slf4j;import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.util.List;/*** @version 1.0* @ClassName MessageCodec* @Description TODO 类描述* @date 2024/4/30  5:46 下午**/
@Slf4j
public class MessageCodec extends ByteToMessageCodec<Message> {/*** 出站前将封装好的Message对象写入到ByteBuf中** @param ctx* @param msg 封装好的消息对象* @param out 写入到消息对象中* @throws Exception*/@Overrideprotected void encode(ChannelHandlerContext ctx, Message msg, ByteBuf out) throws Exception {//1、4个字节的魔数out.writeBytes(new byte[]{1, 2, 3, 4});//2、1个字节版本号:1 表示版本1out.writeByte(1);//3、1个字节序列化算法:0 jdk;1 jsonout.writeByte(0);//4、1个字节指令类型:在Message对象中定义out.writeByte(msg.getMessageType());//5、4个字节:表示请求序号out.writeInt(msg.getSequenceId());//获取内容的字节数组(默认直接采用JDK对象序列化方式)final ByteArrayOutputStream baos = new ByteArrayOutputStream();final ObjectOutputStream oos = new ObjectOutputStream(baos);oos.writeObject(msg);final byte[] data = baos.toByteArray();//(额外):为了满足2的N次方倍,要再加入一个字节凑满16个字节(除实际内容)// 仅仅目的是为了对齐填充out.writeByte(0xff);//6、4个字节length内容长度out.writeInt(data.length);//7、写入内容out.writeBytes(data);}/*** 进行解码:之前怎么封装的就怎么取** @param ctx* @param in* @param out* @throws Exception*/@Overrideprotected void decode(ChannelHandlerContext ctx, ByteBuf in, List<Object> out) throws Exception {final int magicNum = in.readInt();//魔术字final byte version = in.readByte();//版本号final byte serializerType = in.readByte();//序列号final byte messageType = in.readByte();//消息类型final int sequencedId = in.readInt();//请求序号in.readByte();//填充号final int length = in.readInt();//内容长度final byte[] data = new byte[length];
//        in.readBytes(data, 0, length);//内容(字节数组)in.readBytes(data, 0, in.readableBytes());//内容(字节数组)Message message = null;//进行jdk序列化(字节数组转为对象)if (serializerType == 0) {final ObjectInputStream ois = new ObjectInputStream(new ByteArrayInputStream(data));message = (Message) ois.readObject();}out.add(message);log.debug("{}, {}, {}, {}, {}, {}", magicNum, version, messageType, sequencedId);log.debug("{}", message);}}

LoginRequestMessage:

package cn.com.agree.netty.demo.sthalfpackage.protocolAnalysis.custom;/*** @version 1.0* @ClassName LoginRequestMessage* @Description TODO 类描述* @date 2024/5/6  2:44 下午**/
import lombok.Data;
import lombok.ToString;
@Data
@ToString(callSuper = true)
public class LoginRequestMessage extends Message{private String username;private String password;public LoginRequestMessage() {}public LoginRequestMessage(String username, String password) {this.username = username;this.password = password;}@Overridepublic int getMessageType() {return LoginRequestMessage;}
}
2.3.3、TestMessageCodec:测试案例
package cn.com.agree.netty.demo.sthalfpackage.protocolAnalysis.custom;import io.netty.buffer.ByteBuf;
import io.netty.buffer.ByteBufAllocator;
import io.netty.channel.embedded.EmbeddedChannel;
import io.netty.handler.logging.LoggingHandler;
import lombok.extern.slf4j.Slf4j;
import org.junit.Test;import java.util.ArrayList;/*** @version 1.0* @ClassName TestMessageCodec* @Description TODO 类描述* @date 2024/5/6  2:48 下午**/
@Slf4j
public class TestMessageCodec {@Testpublic void  test() throws Exception {EmbeddedChannel embeddedChannel = new EmbeddedChannel(new LoggingHandler(), new MessageCodec());//入站方法测试(编码):encode()LoginRequestMessage loginRequestMessage = new LoginRequestMessage("张三", "1234565");embeddedChannel.writeOutbound(loginRequestMessage);//出站方法测试(解码):decode()ByteBuf byteBuf = ByteBufAllocator.DEFAULT.buffer();//根据协议来进行编码到ByteBuf中new MessageCodec().encode(null,loginRequestMessage,byteBuf);ArrayList<Object> list = new ArrayList<>();//之后按照协议进行的ByteBuf进行解码取得一系列参数及对象new MessageCodec().decode(null,byteBuf,list);log.debug("解码得到的对象是:{}",list.get(0));}}

测试结果:
![在这里插入图片描述](https://img-blog.csdnimg.cn/direct/93375b7c22e8466482e6f410cf5f67bc.png

2.3.4、半包问题出现及解决
  • 半包问题出现原因:若是我们将一个编码过后的ByteBuf分为两个包来入站,那么每发一个包就会走一个decode()也就是解码方法,那么此时可以肯定的是由于包没有发完整,序列化字符串肯定也不完整,那么此时进行解序列化肯定就会报错出现异常!
  • 解决半包思路:我们可以使用LTC解码器来进行解决,按照指定的长度规则来进行解码,那么之前半包会走两次handler再使用了解码器之后,由于半包不完整就会进行等待继续接收包,直到取到完整的包才会走handler那么此时执行decode解码自然不会出现序列化问题!

代码(模拟半包):

/*** 案例2:解码出现半包问题及解决方案,这里仅演示解码情况*  问题描述:若是出现半包问题,那么可能就会出现接解析序列化异常!*  解决方案:使用LTC(基于长度的帧解码器)来解决半包、黏包问题。*/@Testpublic void  test2() throws Exception {EmbeddedChannel embeddedChannel = new EmbeddedChannel(new LoggingHandler(),//使用LTC解码器 12就是用于找到确认正文长度的字节数  4则是表示正文长度数字的字节数
//                new LengthFieldBasedFrameDecoder(1024,12,4,0,0),new MessageCodec());//入站方法测试(编码):encode()LoginRequestMessage loginRequestMessage = new LoginRequestMessage("张三", "1234565");embeddedChannel.writeOutbound(loginRequestMessage);//出站方法测试(解码):decode()ByteBuf buffer = ByteBufAllocator.DEFAULT.buffer();//根据协议来进行编码到ByteBuf中new MessageCodec().encode(null,loginRequestMessage,buffer);//进行切片将数据切分成两片ByteBuf firSlice = buffer.slice(0, 100);ByteBuf secSlice = buffer.slice(100, buffer.readableBytes() - 100);//模拟入站操作,此时就会执行decode方法embeddedChannel.writeInbound(firSlice);ArrayList<Object> list = new ArrayList<>();//之后按照协议进行的ByteBuf进行解码取得一系列参数及对象new MessageCodec().decode(null,buffer,list);log.debug("解码得到的对象是:{}",list.get(0));}

结果:解析异常
在这里插入图片描述
加上LTC解码器 解决半包问题
代码:

 /*** 案例2:解码出现半包问题及解决方案,这里仅演示解码情况*  问题描述:若是出现半包问题,那么可能就会出现接解析序列化异常!*  解决方案:使用LTC(基于长度的帧解码器)来解决半包、黏包问题。*/@Testpublic void  test2() throws Exception {EmbeddedChannel embeddedChannel = new EmbeddedChannel(new LoggingHandler(),//使用LTC解码器 12就是用于找到确认正文长度的字节数  4则是表示正文长度数字的字节数new LengthFieldBasedFrameDecoder(1024,12,4,0,0),new MessageCodec());//入站方法测试(编码):encode()LoginRequestMessage loginRequestMessage = new LoginRequestMessage("张三", "1234565");embeddedChannel.writeOutbound(loginRequestMessage);//出站方法测试(解码):decode()ByteBuf buffer = ByteBufAllocator.DEFAULT.buffer();//根据协议来进行编码到ByteBuf中new MessageCodec().encode(null,loginRequestMessage,buffer);//进行切片将数据切分成两片ByteBuf firSlice = buffer.slice(0, 100);ByteBuf secSlice = buffer.slice(100, buffer.readableBytes() - 100);//模拟入站操作,此时就会执行decode方法embeddedChannel.writeInbound(firSlice);ArrayList<Object> list = new ArrayList<>();buffer.retain();//引用计数加1embeddedChannel.writeInbound(secSlice);//执行一次writeInbound实际上就会执行release()进行内存释放,由于这里为了避免释放,自此之前引用计数+1}

结果:
在这里插入图片描述

2.4、@Sharable:可共享handler
  • 我们之前编写代码时都会在初始化channel时添加执行器handler,添加的方式如下,我们可以看到每次都是添加到channel的pipeline管道对象中,那么就会有一个疑问:每个channel在创建之初都会添加新的handler到管道中吗?
//实际写服务端的handler
.childHandler(new ChannelInitializer<NioSocketChannel>() {@Overrideprotected void initChannel(NioSocketChannel ch) throws Exception {ch.pipeline().addLast(new LoggingHandler())ch.pipeline().addLast(new ChannelInboundHandlerAdapter(){}}
}//EmbeddedChannel添加handler方式
final EmbeddedChannel channel = new EmbeddedChannel(new LoggingHandler(),new LengthFieldBasedFrameDecoder(1024, 12, 4, 0, 0),new MessageCodec()   //自定义的编解码器
);

在这里插入图片描述

  • 通过debug测试可以得出结论:若是我们直接在addLast()中添加新的handler,那么就是每初始化一个连接就会创建多个新的handler对象!
  • 那么我们可以想到一个优化思路,既然如此那么为什么我们不提前直接new出来一个handler作为多个channel的共享执行器?对的确实可以,对于我们自定义的一些执行器我们可以清楚的知道有没有使用一些共享对象,有没有线程安全问题,而对于netty提供给我们的我们不能够确定,这时候nettty给了我们思路来看其提供的handler是否是可共享、是线程安全的:实际上就是@Sharable。
  • 结论:如果netty提供给我们的handler是可共享的,就会标注该类为@Sharable,我们可以放心大胆的创建一个公共handler来被多个channel共同使用!
  • 对于LoggingHandler是有该注解的,表示是可共享的!
    在这里插入图片描述
  • ByteToMessageCodec的子类无法添加@Sharable,使用会报错 在这里插入图片描述
  • 分析:netty框架设计者在设计ByteToMessageCodec类时设定就是实现该类的一些编解码器正常来说都会使用到一个公共变量存储及操作,其认为本该是不可共享的,如下代码注释说明,该类不允许被标注
    在这里插入图片描述
  • 解决方案:那么若是我们自定义的编解码器确实没有使用到公共变量,我们想将其标注为
    @Sharable呢?可以将原本实现ByteToMessageCodec更改MessageToMessageCodec
package cn.com.agree.netty.demo.sthalfpackage.protocolAnalysis.custom;/*** @version 1.0* @ClassName MessageCodecSharable* @Description TODO 类描述* @date 2024/5/6  5:31 下午**/import io.netty.buffer.ByteBuf;
import io.netty.channel.ChannelHandler;
import io.netty.channel.ChannelHandlerContext;
import io.netty.handler.codec.MessageToMessageCodec;
import lombok.extern.slf4j.Slf4j;import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.util.List;
@Slf4j
@ChannelHandler.Sharable
public class MessageCodecSharable extends MessageToMessageCodec<ByteBuf, Message> {@Overrideprotected void encode(ChannelHandlerContext ctx, Message msg, List<Object> out) throws Exception {final ByteBuf buffer = ctx.alloc().buffer();//1、4个字节的魔数buffer.writeBytes(new byte[]{1,2,3,4});//2、1个字节版本号:1 表示版本1buffer.writeByte(1);//3、1个字节序列化算法:0 jdk;1 jsonbuffer.writeByte(0);//4、1个字节指令类型:在Message对象中定义buffer.writeByte(msg.getMessageType());//5、4个字节:表示请求序号buffer.writeInt(msg.getSequenceId());//获取内容的字节数组(默认直接采用JDK对象序列化方式)final ByteArrayOutputStream baos = new ByteArrayOutputStream();final ObjectOutputStream oos = new ObjectOutputStream(baos);oos.writeObject(msg);final byte[] data = baos.toByteArray();//(额外):为了满足2的N次方倍,要再加入一个字节凑满16个字节(除实际内容)// 仅仅目的是为了对齐填充buffer.writeByte(0xff);//6、4个字节length内容长度buffer.writeInt(data.length);//7、写入内容buffer.writeBytes(data);out.add(buffer);}@Overrideprotected void decode(ChannelHandlerContext ctx, ByteBuf in, List<Object> out) throws Exception {final int magicNum = in.readInt();//魔术字final byte version = in.readByte();//版本号final byte serializerType = in.readByte();//序列号final byte messageType = in.readByte();//消息类型final int sequencedId = in.readInt();//请求序号in.readByte();//填充号final int length = in.readInt();//内容长度final byte[] data = new byte[length];
//        in.readBytes(data, 0, length);//内容(字节数组)in.readBytes(data, 0, in.readableBytes());//内容(字节数组)Message message = null;//进行jdk序列化(字节数组转为对象)if (serializerType == 0) {final ObjectInputStream ois = new ObjectInputStream(new ByteArrayInputStream(data));message = (Message) ois.readObject();}out.add(message);log.debug("{}, {}, {}, {}, {}, {}", magicNum, version, messageType, sequencedId);log.debug("{}", message);}
}

server端改进后的代码:

/*** 案例2:解码出现半包问题及解决方案,这里仅演示解码情况*  问题描述:若是出现半包问题,那么可能就会出现接解析序列化异常!*  解决方案:使用LTC(基于长度的帧解码器)来解决半包、黏包问题。*/@Testpublic void  test2() throws Exception {LoggingHandler loggingHandler = new LoggingHandler();MessageCodecSharable messageCodecSharable = new MessageCodecSharable();EmbeddedChannel embeddedChannel = new EmbeddedChannel(loggingHandler,//使用LTC解码器 12就是用于找到确认正文长度的字节数  4则是表示正文长度数字的字节数new LengthFieldBasedFrameDecoder(1024,12,4,0,0),messageCodecSharable);//入站方法测试(编码):encode()LoginRequestMessage loginRequestMessage = new LoginRequestMessage("张三", "1234565");embeddedChannel.writeOutbound(loginRequestMessage);//出站方法测试(解码):decode()ByteBuf buffer = ByteBufAllocator.DEFAULT.buffer();//根据协议来进行编码到ByteBuf中new MessageCodec().encode(null,loginRequestMessage,buffer);//进行切片将数据切分成两片ByteBuf firSlice = buffer.slice(0, 100);ByteBuf secSlice = buffer.slice(100, buffer.readableBytes() - 100);//模拟入站操作,此时就会执行decode方法embeddedChannel.writeInbound(firSlice);ArrayList<Object> list = new ArrayList<>();buffer.retain();//引用计数加1embeddedChannel.writeInbound(secSlice);//执行一次writeInbound实际上就会执行release()进行内存释放,由于这里为了避免释放,自此之前引用计数+1}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/670230.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【前端】输入时字符跳动动画实现

输入时字符跳动动画实现 在前端开发中&#xff0c;为了提升用户体验&#xff0c;我们经常需要为用户的交互行为提供即时的反馈。这不仅让用户知道他们的操作有了响应&#xff0c;还可以让整个界面看起来更加生动、有趣。本文将通过一个简单的例子讲解如何实现在用户输入字符时…

MySQL#MySql数据库的操作

目录 一、创建数据库 二、字符集和校验规则 1.查看系统默认字符集以及校验规则 2.查看数据库支持的字符集 3.查看数据库支持的字符集校验规则 4.校验规则对数据库的影响 1.以UTF-8格式创建数据库 2.不区分大小写 3.区分大小写 4 大小写对数据库的影响 三、操纵数据…

SLAIM:一个实时的RGB-D NeRF-SLAM系统

SLAIM&#xff1a;一个实时的RGB-D NeRF-SLAM系统与现有的NeRF-SLAM系统相比&#xff0c;我们的方法在跟踪性能上始终表现出更强的竞争力。我们的方法采用体积密度表示&#xff0c;并引入了一种新的KL正则化器在射线终止分布上&#xff0c;将场景几何限制为空隙空间和不透明表面…

数据处理学习笔记9

一些其他的函数 “Resize”和“Reshape”的区别主要在于它们对数组元素数量和形状的处理方式不同&#xff0c;以下是详细介绍&#xff1a; “Resize”通常会改变数组的元素数量&#xff0c;在放大数组形状时会用0补全新增的元素&#xff0c;而在缩小数组形状时会丢弃多余的元素…

Android 状态栏WiFi图标的显示逻辑

1. 状态栏信号图标 1.1 WIFI信号显示 WIFI信号在状态栏的显示如下图所示 当WiFi状态为关闭时&#xff0c;状态栏不会有任何显示。当WiFi状态打开时&#xff0c;会如上图所示&#xff0c;左侧表示有可用WiFi&#xff0c;右侧表示当前WiFi打开但未连接。 当WiFi状态连接时&#x…

深入C语言:文件操作实现局外影响程序

一、什么是文件 文件其实是指一组相关数据的有序集合。这个数据集有一个名称&#xff0c;叫做文件名。文件通常是驻留在外部介质(如磁盘等)上的&#xff0c;在使用时才调入内存中来。 文件一般讲两种&#xff1a;程序文件和数据文件&#xff1a; 程序文件&#xff1a;包括源程…

StreamingT2V

下面首先是参考的一些博客 https://blog.csdn.net/qq_44681809/article/details/137081515 qustion SDEdit:就是给图片加一点噪声然后再用模型去噪&#xff0c;来获得一个更好的帧&#xff0c;比如去掉伪影和污点 这里的分割为m个24帧的块&#xff0c;块与块之间已经有8帧重叠…

04-18 周四 为LLM_inference项目配置GitHub CI过程记录

04-18 周四 为LLM_inference项目配置GitHub CI过程记录 时间版本修改人描述2024年4月18日10:30:13V0.1宋全恒新建文档 简介和相关文档 04-15 周一 GitHub仓库CI服务器配置过程文档actions-runner 是托管与GitHub上的仓库&#xff0c;下载最新的客户端程序即可。self hosted r…

前端css中animation(动画)的使用

前端css中animation的使用 一、前言二、主要内容说明&#xff08;一&#xff09;、animation-name&#xff08;名称&#xff09;属性&#xff08;二&#xff09;、animation-duration&#xff08;持续时间&#xff09;属性1.前两个属性举例&#xff0c;源码12.源码1运行效果&am…

【UE5学习笔记】编辑及运行界面:关闭眼部识别(自动曝光)

自动曝光&#xff0c;也就是走进一个黑暗的环境&#xff0c;画面会逐渐变量&#xff0c;以模拟人眼进入黑暗空间时瞳孔放大&#xff0c;进光量增加的一种真实视觉感受&#xff1a; 制作过程中是否关闭自动曝光&#xff0c;取决于游戏的性质&#xff0c;但是个人认为&#xff0c…

【一刷《剑指Offer》】面试题 14:调整数组顺序使奇数位于偶数前面

力扣对应题目链接&#xff1a;LCR 139. 训练计划 I - 力扣&#xff08;LeetCode&#xff09; 牛客对应题目链接&#xff1a;调整数组顺序使奇数位于偶数前面(二)_牛客题霸_牛客网 (nowcoder.com) 核心考点&#xff1a;数组操作&#xff0c;排序思想的扩展使用。 一、《剑指Off…

更适合宝妈和上班族的兼职,每天2小时收入250+的微头条项目

许多人通过撰写微头条赚取收入&#xff0c;但这通常需要自己寻找素材&#xff0c;然后逐字逐句地进行改编创作&#xff0c;整个过程既繁琐又低效。 然而&#xff0c;如今全球范围内的AI工具正如雨后春笋般涌现。百度推出了文心一言&#xff0c;阿里巴巴推出了AI工具通义千问&a…