netty中收到的消息是ByteBuf类型,需要先将其转换为java对象,才能做业务处理。发出消息时则需要先将java对象转为ByteBuf。解码器和编码器就是专门用来处理这种转换的,收到消息时解码,发出消息时编码。
启动配置
修改netty启动代码,现在有三个自定义类,多了解码器和编码器。解码器是入站处理,编码器是出站处理。
ServerBootstrap b = new ServerBootstrap();
b.group(bossGroup, workerGroup).channel(NioServerSocketChannel.class)
.childHandler(new ChannelInitializer<SocketChannel>() {
@Override
public void initChannel(SocketChannel ch) throws Exception {
ch.pipeline().addLast(new MyDecoder());//解码器
ch.pipeline().addLast(new MyEncoder());//编码器
ch.pipeline().addLast(new MyHandler());//业务处理
}
}).option(ChannelOption.SO_BACKLOG, 128)
.childOption(ChannelOption.SO_KEEPALIVE, true);
解码
继承MessageToMessageDecoder即可,在这里将收到的ByteBuf消息转换成字符串。可能你读取到的是一个json字符串,你可以在这里转换成自定义java类型,然后放入out列表中。如果在out中放入了多个对象,每个对象都会单独触发一次业务处理。
public class MyDecoder extends MessageToMessageDecoder<ByteBuf>{
@Override
protected void decode(ChannelHandlerContext ctx, ByteBuf msg, List<Object> out) throws Exception {
String s = msg.toString(CharsetUtil.UTF_8);
out.add(s);
}
业务处理
这里String类型msg是不能直接发出去的,还需要有编码器码转换成ByteBuf 才能发出。
public class MyHandler extends SimpleChannelInboundHandler<String> {
protected void channelRead0(ChannelHandlerContext ctx, String msg) {
System.out.println(msg);
ctx.writeAndFlush(msg);
}
编码
把msg字符串写入ByteBuf,加入out列表中,就可以正常发出消息了。这里的List和解码时一样,加入多个不同的ByteBuf会发出多条消息。
public class MyEncoder extends MessageToMessageEncoder<String>{
@Override
protected void encode(ChannelHandlerContext ctx, String msg, List<Object> out) throws Exception {
ByteBuf b = Unpooled.buffer();
b.writeBytes(msg.getBytes());
out.add(b);
}
预设解码和编码
netty中有一些预设的解码和编码类,比如字符串解码StringDecoder和字符串编码StringEncoder,可以直接使用。启动类做以下修改,效果和上面是一样的。
ch.pipeline().addLast(new StringDecoder());
ch.pipeline().addLast(new StringEncoder());
ch.pipeline().addLast(new MyHandler());
字节解码
socket项目的很多场景,如物联网,需要自己去逐个读取16进制字节,根据每个字节的含义,分别给自定义的java类型参数赋值。此时则使用ByteToMessageDecoder解码,逐字读取和16进制用法参考第3篇。
public class MyDecoder extends ByteToMessageDecoder{
@Override
protected void decode(ChannelHandlerContext ctx, ByteBuf in, List<Object> out) throws Exception {
ByteBuf b = in.readBytes(in.readableBytes());
String msg = b.toString(CharsetUtil.UTF_8);
out.add(msg);
}
字节编码
字节编码和MessageToMessageEncoder类似,但一次只能发出一条消息。
public class MyEncoder extends MessageToByteEncoder<String>{
@Override
protected void encode(ChannelHandlerContext ctx, String msg, ByteBuf out) throws Exception {
out.writeBytes(msg.getBytes());
}