Netty (6)-解码和编码

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());
	}

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值