Netty Handler的调用机制

23 篇文章 2 订阅
NettyLongToByteEncoder:
import io.netty.buffer.ByteBuf;
import io.netty.channel.ChannelHandlerContext;
import io.netty.handler.codec.MessageToByteEncoder;

public class NettyLongToByteEncoder extends MessageToByteEncoder<Long> {
    @Override
    protected void encode(ChannelHandlerContext channelHandlerContext, Long aLong, ByteBuf byteBuf) throws Exception {
        System.out.println("NettyLongToByteEncoder encode 被调用");
        System.out.println("msg = " + aLong);
        byteBuf.writeLong(aLong);
    }
}
NettyByteToLongDecoder:
import io.netty.buffer.ByteBuf;
import io.netty.channel.ChannelHandlerContext;
import io.netty.handler.codec.ByteToMessageDecoder;

import java.util.List;

public class NettyByteToLongDecoder extends ByteToMessageDecoder {
    /**
     * abcdabcdabcdabcd 会根据接收到的数据,被调用多次, 知道确定没有新的元素被添加到list 或者 ByteBuf 没有更多的可读字节为止
     * 如果 list 的不为空,就会将 list 的内容传递给下一个 channelInboundhandler处理,该处理器的方法也会被调用多次
     * @param channelHandlerContext 上下文对象
     * @param byteBuf 入站的 byteBuf
     * @param list 将解码后的数据传递给下一个 handler
     * @throws Exception
     */
    @Override
    protected void decode(ChannelHandlerContext channelHandlerContext, ByteBuf byteBuf, List<Object> list) throws Exception {
        System.out.println("NettyByteToLongDecoder 被调用");
        // 因为 Long 8个字节
        if (byteBuf.readableBytes() >= 8) {
            list.add(byteBuf.readLong());
        }
    }
}
NettyServer:

import io.netty.bootstrap.ServerBootstrap;
import io.netty.channel.ChannelFuture;
import io.netty.channel.nio.NioEventLoopGroup;
import io.netty.channel.socket.nio.NioServerSocketChannel;

public class NettyServer {
    public static void main(String[] args) throws InterruptedException {
        NioEventLoopGroup bossGroup = new NioEventLoopGroup(1);
        NioEventLoopGroup workerGroup = new NioEventLoopGroup();

        try {
            ServerBootstrap serverBootstrap = new ServerBootstrap();
            serverBootstrap.group(bossGroup, workerGroup)
                    .channel(NioServerSocketChannel.class)
                    .childHandler(new NettyServerInitializer());

            ChannelFuture channelFuture = serverBootstrap.bind(9527).sync();
            channelFuture.channel().closeFuture().sync();
        } finally {
            bossGroup.shutdownGracefully();
            workerGroup.shutdownGracefully();
        }

    }
}
NettyServerHandler:
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.SimpleChannelInboundHandler;

public class NettyServerHandler extends SimpleChannelInboundHandler<Long> {
    @Override
    protected void channelRead0(ChannelHandlerContext channelHandlerContext, Long aLong) throws Exception {
        System.out.println("从客户端" + channelHandlerContext.channel().remoteAddress() + " 读取到Long" + aLong);

        // 给客户端发送一个 Long
        channelHandlerContext.writeAndFlush(4396L);
    }


    @Override
    public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) throws Exception {
        cause.printStackTrace();
        ctx.close();
    }
}
NettyServerInitializer:
import io.netty.channel.ChannelInitializer;
import io.netty.channel.ChannelPipeline;
import io.netty.channel.socket.SocketChannel;

public class NettyServerInitializer extends ChannelInitializer<SocketChannel> {
    @Override
    protected void initChannel(SocketChannel socketChannel) throws Exception {
        ChannelPipeline pipeline = socketChannel.pipeline();

        // 入站的 handler 进行解码
        pipeline.addLast(new NettyByteToLongDecoder());
        // 出站的 handler 进行编码,  入站 和 出站 两条线进行走
        pipeline.addLast(new NettyLongToByteEncoder());
        // 自定义的 handler 处理业务逻辑
        pipeline.addLast(new NettyServerHandler());
    }
}
NettyClient:
import io.netty.bootstrap.Bootstrap;
import io.netty.channel.ChannelFuture;
import io.netty.channel.EventLoopGroup;
import io.netty.channel.nio.NioEventLoopGroup;
import io.netty.channel.socket.nio.NioSocketChannel;

public class NettyClient {

    public static void main(String[] args) throws InterruptedException {
        EventLoopGroup group = new NioEventLoopGroup();
        try {
            Bootstrap bootstrap = new Bootstrap();
            bootstrap.group(group).channel(NioSocketChannel.class).handler(new NettyClientInitializer());

            ChannelFuture channelFuture = bootstrap.connect("localhost", 9527).sync();
            channelFuture.channel().closeFuture().sync();
        }finally {
            group.shutdownGracefully();
        }

    }
}
NettyClientHandler:
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.SimpleChannelInboundHandler;

public class NettyClientHandler extends SimpleChannelInboundHandler<Long> {

    @Override
    protected void channelRead0(ChannelHandlerContext channelHandlerContext, Long aLong) throws Exception {
        System.out.println("服务器的ip=" + channelHandlerContext.channel().remoteAddress());
        System.out.println("收到服务器消息=" + aLong);
    }


    @Override
    public void channelActive(ChannelHandlerContext ctx) throws Exception {
        System.out.println("NettyClientHandler 发送数据");

         ctx.writeAndFlush(9527L);
        /**
         * abcdabcdabcd 16个字节
         * 该处理器的前一个 handler 是 NettyLongToByteEncoder
         * NettyLongToByteEncoder 的父类是 MessageToByteEncoder
         * 在编写 encoder 时, 要注意传入的数据类型和处理的数据类型一致
         */
        //ctx.writeAndFlush(Unpooled.copiedBuffer("abcdabcdabcdabcd", CharsetUtil.UTF_8));
    }
}
NettyClientInitializer:
import io.netty.channel.ChannelInitializer;
import io.netty.channel.ChannelPipeline;
import io.netty.channel.socket.SocketChannel;

public class NettyClientInitializer extends ChannelInitializer<SocketChannel> {
    @Override
    protected void initChannel(SocketChannel socketChannel) throws Exception {
        ChannelPipeline pipeline = socketChannel.pipeline();

        // 加入一个 出站的 handler 对数据进行编码
        pipeline.addLast(new NettyLongToByteEncoder());

        // 这是一个入站的解码器(handler)
        pipeline.addLast(new NettyByteToLongDecoder());

        // 加入一个自定义的 handler, 处理业务逻辑
        pipeline.addLast(new NettyClientHandler());
    }
}
MessageToByteEncoder 的 write 方法:
public void write(ChannelHandlerContext ctx, Object msg, ChannelPromise promise) throws Exception {
        ByteBuf buf = null;

        try {
            if (this.acceptOutboundMessage(msg)) { // 判断当前 msg, 是不是该处理的类型
                I cast = msg;
                buf = this.allocateBuffer(ctx, msg, this.preferDirect);

                try {
                    this.encode(ctx, cast, buf);
                } finally {
                    ReferenceCountUtil.release(msg);
                }

                if (buf.isReadable()) {
                    ctx.write(buf, promise);
                } else {
                    buf.release();
                    ctx.write(Unpooled.EMPTY_BUFFER, promise);
                }

                buf = null;
            } else { // 直接发送
                ctx.write(msg, promise);
            }
        } catch (EncoderException var17) {
            throw var17;
        } catch (Throwable var18) {
            throw new EncoderException(var18);
        } finally {
            if (buf != null) {
                buf.release();
            }

        }

    }

结论:

  • 不论解码器 handler 还是 编码器 handler 即接收的消息类型必须与待处理消息类型一致,否则该 handler 不会被执行
  • 在解码器 进行数据解码时,需要判断缓存区(ByteBuf)的数据是否足够,否则接收到的结果会与期望结果可能不一致

Netty HandlerNetty 框架中用于处理网络数据包的核心组件。它是一个 Java 接口,所有处理网络请求的逻辑通常都是通过这个接口来完成的。Handler 的设计允许开发者自定义各种处理流程,包括读取、解码、业务处理、编码、写入等步骤。 ### 手册的基本组成部分: #### ChannelHandlerContext `ChannelHandlerContext` 是 `Handler` 类型的引用,并且包含了当前通道上下文信息。这意味着它可以访问通道本身以及在其生命周期内的事件监听器。在编写处理器时,经常需要处理诸如 `read()`、`write()` 或 `channelInactive()` 等通道事件。 #### 生命周期管理 每个 `Handler` 实现都有其特定的生命周期阶段,在这些阶段内,它们可以响应不同的通道事件。例如,当接收新连接时,某个特定 `Handler` 可能会被初始化;在网络连接关闭时,另一个 `Handler` 可能会被通知并采取相应的清理措施。 #### 链式结构 Netty 使用链式的处理结构,即多个 `Handler` 可以串联在一起形成一条“处理链”。这使得数据流能够按照预设的顺序经过一系列的处理操作。这种设计便于模块化地构建复杂的网络应用逻辑,同时也简化了错误追踪和调试过程。 ### 应用场景及示例 在实际应用中,Netty Handlers 可用于多种场景,如 TCP 和 UDP 协议处理、HTTP 请求解析、WebSocket 协议支持等。例如,对于一个 HTTP 服务器应用,你可以创建一个 Handler 来解析 HTTP 请求行、头部字段,然后调用适当的业务逻辑,最后生成响应并编码成 HTTP 格式发送回客户端。 ```java public class HttpRequestHandler extends ChannelInboundHandlerAdapter { @Override public void channelRead(ChannelHandlerContext ctx, Object msg) throws Exception { // 解析 HTTP 请求消息 HttpRequest request = (HttpRequest) msg; String method = request.getMethod(); if ("GET".equals(method)) { // 处理 GET 请求,这里只是一个简单的示例 ctx.writeAndFlush(new DefaultFullHttpResponse(HttpVersion.HTTP_1_1, HttpResponseStatus.OK)); } } // 这里可以添加其他 Channel Lifecycle 方法的覆盖实现 } ``` ### 相关问题: 1. **如何创建并配置 Netty Handler**? 创建一个 Netty Handler 通常涉及到继承 `ChannelInboundHandlerAdapter` 或者更具体的 `ChannelHandler` 接口,并在其中实现 `channelRead` 等方法。配置方面,可以通过设置处理器链、调整处理器的位置和优先级来控制消息的处理顺序。 2. **为什么 Netty 使用链式处理模型**? Netty 使用链式处理模型的主要原因是它提供了高度的灵活性和可组合性。这种设计允许开发者以模块化的方式组织和扩展功能,同时易于管理和测试,减少了耦合度。 3. **常见的 Netty Handler 示例有哪些**? 常见的 Netty Handler 包括用于数据编码/解码的处理器(如 `BinaryDecoder` 和 `BinaryEncoder`)、用于日志记录的处理器 (`LoggingHandler`)、用于异常处理的处理器(如 `ExceptionLoggerHandler`),以及用于协议解析的处理器(如用于 WebSocket、HTTP 等)。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值