1 Netty服务端

  第一章提出“Netty作为服务端启动时偶尔会有服务端意外退出的现象”的问题。原因:Netty作为服务端启动,为了防止服务端意外退出,应该防止EventLoopGroup意外关闭,因为EventLoopGroup是非守护线程,只要它没有退出,则JVM不会关闭。
  若启动线程是main线程,且优雅关闭作业也在main线程,则启动完成后需要阻塞线程,否则finally块会关闭EventLoopGroup:

正确的用法示例:
        EventLoopGroup executors = new NioEventLoopGroup(3);
        ServerBootstrap bootstrap = new ServerBootstrap();

        bootstrap.group(executors)
                .option(ChannelOption.SO_BACKLOG, 1024)
                .childOption(ChannelOption.SO_KEEPALIVE, true)
                .channel(NioServerSocketChannel.class)
                .localAddress(serverPort)
                .childHandler(new NATChannelInitializer(true));

        try {
            ChannelFuture future = bootstrap.bind().sync();
            future.channel().closeFuture().sync(); // 阻塞main线程

        } catch (InterruptedException e) {
            e.printStackTrace();

        } finally {
            executors.shutdownGracefully();
        }
错误的用法示例:
        EventLoopGroup executors = new NioEventLoopGroup(3);
        ServerBootstrap bootstrap = new ServerBootstrap();

        bootstrap.group(executors)
                .option(ChannelOption.SO_BACKLOG, 1024)
                .childOption(ChannelOption.SO_KEEPALIVE, true)
                .channel(NioServerSocketChannel.class)
                .localAddress(serverPort)
                .childHandler(new NATChannelInitializer(true));

        try {
            ChannelFuture future = bootstrap.bind().sync();
            // 没有阻塞main线程,程序继续执行到finally块,导致EventLoopGroup关闭,服务端意外退出

        } catch (InterruptedException e) {
            e.printStackTrace();

        } finally {
            executors.shutdownGracefully();
        }

  若启动线程不是main线程,即启动线程由tomcat或spring boot某个方法调用启动,则这时不应该阻塞线程,应该让调用立即返回,优雅关闭作业可以放在Channel的CloseListener中。

错误的用法示例(同样的代码,在main线程中是正确的,在方法调用中是错误的):
        EventLoopGroup executors = new NioEventLoopGroup(3);
        ServerBootstrap bootstrap = new ServerBootstrap();

        bootstrap.group(executors)
                .option(ChannelOption.SO_BACKLOG, 1024)
                .childOption(ChannelOption.SO_KEEPALIVE, true)
                .channel(NioServerSocketChannel.class)
                .localAddress(serverPort)
                .childHandler(new NATChannelInitializer(true));

        try {
            ChannelFuture future = bootstrap.bind().sync();
            future.channel().closeFuture().sync(); // 阻塞了调用线程,调用不能立即返回

        } catch (InterruptedException e) {
            e.printStackTrace();
        }
正确的示例:
        EventLoopGroup executors = new NioEventLoopGroup(3);
        ServerBootstrap bootstrap = new ServerBootstrap();

        bootstrap.group(executors)
                .option(ChannelOption.SO_BACKLOG, 1024)
                .childOption(ChannelOption.SO_KEEPALIVE, true)
                .channel(NioServerSocketChannel.class)
                .localAddress(serverPort)
                .childHandler(new NATChannelInitializer(true));

        try {
            ChannelFuture future = bootstrap.bind().sync();
            // 添加监听器执行优雅关闭作业
            future.channel().closeFuture().addListener((futureX)->{
                 executors.shutdownGracefully();
            })
            // 不阻塞调用线程,调用立即返回

        } catch (InterruptedException e) {
            e.printStackTrace();
        }

2 优雅关闭

  Netty优雅关闭作业shutdownGracefully(),该方法位于EventExecutorGroup接口中,在netty-4.1.36.Final,EventLoop(单线程线程池SingleThreadEventExecutor)和EventLoopGroup(多线程线程池MultithreadEventExecutorGroup)都有实现。关闭过程中的工作包括:(1)设置线程状态码(2)发送尚未发送或正在发送的消息(3)完成到期的定时任务(4)执行NIO线程的Hook(5)Channel释放,剩余定时任务释放,多路复用器注销,退出EventLoop线程。

3 Netty客户端

  第二章提出“客户端连接池资源泄露,出现OOM错误”。原因:错误的用法是每初始化一个TCP连接时初始化了一个EventLoopGroup,EventLoopGroup资源占用比EventLoop大,然而一个EventLoopGroup可以管理不止一个TCP连接(EventLoop)。Netty作为客户端实际上是把Netty作为TCP连接池使用,Redis的Java客户端Redisson就使用了Netty作为连接池,可以参考其用法。

正确的用法示例:
// 客户端TCP连接集合
private List<ChannelFuture> channelFutures = new ArrayList<ChannelFuture>();

public void start() {
        EventLoopGroup executors = new NioEventLoopGroup();
        try {
            Bootstrap bootstrap = new Bootstrap();
            bootstrap.option(ChannelOption.SO_KEEPALIVE, true);
            bootstrap.option(ChannelOption.SO_REUSEADDR, true);
            bootstrap.group(executors)
                    .channel(NioSocketChannel.class)
                    .remoteAddress(new InetSocketAddress(serverHost, serverPort))
                    .handler(new NATChannelInitializer(false));
            //初始化多个TCP连接
            for (int i = 0; i < 10; i++) {
                this.channelFutures.add(bootstrap.connect().sync());
            }

        } catch (InterruptedException e) {
            e.printStackTrace();
        }
}
错误的用法示例:
// 客户端TCP连接集合
private List<ChannelFuture> channelFutures = new ArrayList<ChannelFuture>();

public void start() {
    //初始化多个TCP连接
    for (int i = 0; i < 10; i++) {
        EventLoopGroup executors = new NioEventLoopGroup();
        try {
            Bootstrap bootstrap = new Bootstrap();
            bootstrap.option(ChannelOption.SO_KEEPALIVE, true);
            bootstrap.option(ChannelOption.SO_REUSEADDR, true);
            bootstrap.group(executors)
                    .channel(NioSocketChannel.class)
                    .remoteAddress(new InetSocketAddress(serverHost, serverPort))
                    .handler(new NATChannelInitializer(false));
            
                this.channelFutures.add(bootstrap.connect().sync());
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}

总结

  第一、二章介绍了Netty作为服务端和客户端的正确用法,顺带介绍了Netty的优雅关闭。