【Netty4】future.channel().closeFuture().sync()作用& bind(port).sync()作用

1. 实例

我们看下官方的入门例子:

启动一个nettyserver,监听8088端口,再通过命令行执行 telnet localhost 8088来连接。
完整的例子请参考之前的入门例子介绍:
Netty入门官方例子解析(一)丢弃服务器《入门例子,不返回消息》章节

public class DiscardServer {       
        try {                  
            //bind操作(对应初始化)是异步的,通过sync改为同步等待初始化的完成,否则立即操作对象(未初始完全)可能会报错
            f = b.bind(port).sync();
            // Wait until the server socket is closed.
            // In this example, this does not happen, but you can do that to gracefully
            // shut down your server.
            f.channel().closeFuture().sync();
        } finally {
            // 资源优雅释放
            bossGroup.shutdownGracefully();
            workerGroup.shutdownGracefully();
            f.channel().close();
        }
    }   
}

在这里面future.channel().closeFuture().sync();这个语句的主要目的是,如果缺失上述代码,则main方法所在的线程,即主线程会在执行完bind().sync()方法后,会进入finally 代码块,之前的启动的nettyserver也会随之关闭掉,整个程序都结束了。

f = b.bind(port).sync();也有阻塞功能,为啥还要后面的那个阻塞?答案是这里的阻塞很快会结束,只会阻塞一会,一旦连接器初始化完成,就会结束阻塞。

原文的例子有英文注释:
Wait until the server socket is closed,In this example, this does not happen, but you can do that to gracefully shut down your server.
让线程进入wait状态,也就是main线程暂时不会执行到finally里面,nettyserver也持续运行,如果监听到关闭事件,可以优雅的关闭通道和nettyserver,虽然这个例子中,永远不会监听到关闭事件(代码比较简洁,没有涉及触发关闭事件的代码)。也就是说这个例子是仅仅为了展示存在api shutdownGracefully,可以优雅的关闭nettyserver。

当然,如果你在代码中通过其他线程触发了关闭事件,此时main会响应事件,并进入finally代码块,进行优雅的关闭操作。

下面我们来验证下:

public class DiscardServer {
       
        try {
                       
            f = b.bind(port).sync();
          //  f.channel().closeFuture().sync();
        } finally {
            // 资源优雅释放
            bossGroup.shutdownGracefully();
            workerGroup.shutdownGracefully();
            f.channel().close();
        }
    }   
}

我们注掉了future.channel().closeFuture().sync(),这时,再执行telnet命令的时候会提示连接失败
在这里插入图片描述

如果我们不想加f.channel().closeFuture().sync()又想保证程序正常运行怎么办,简单,去掉finally 里面关闭nettyserver的语句即可。下面我们来改造下:

 try {
                       
            f = b.bind(port).sync();
          //  f.channel().closeFuture().sync();
        } finally {
            // 资源优雅释放
            //bossGroup.shutdownGracefully();
            //workerGroup.shutdownGracefully();
            //f.channel().close();
        }

是不是一目了然了?其实入门例子可以完全按照我的改造的方法来演示,去掉f.channel().closeFuture().sync()。否则,对于刚入门的小白来说,无法理解这个例子中的奥义。

2. 深入研究

前文得知,future.channel().closeFuture().sync()作用是为了让netty不会关闭,那么具体是什么原理呢?

原因是这样:netty为了基于性能,多采用异步方式,而我们通过调用sync()方法,会让主线程间接调用wait()方法,进而实现阻塞的效果。

当然,在简单的demo中,会这样实现阻塞,但是在正式环境下,一般不会这么写。一般是通过信号量或其他机制,监听关闭事件,然后进行调用 ,信号量 用法 参见 如何优雅地停止Java进程

同样的,在前面“b.bind(port).sync();”中也有sync(),原理类似,当调用bind(port)时,是异步的,因此为了保证在初始化完成后才进行操作,避免调用一个初始化未完成的句柄,sync方法是等待异步操作执行完毕.

  • 22
    点赞
  • 29
    收藏
    觉得还不错? 一键收藏
  • 8
    评论
这段代码使用了Netty框架来启动一个服务器并绑定多个端口。下面是对代码的解析和问题的检查: 1. 首先,创建了两个`NioEventLoopGroup`对象,分别用于处理服务器的连接请求(bossGroup)和处理已经建立连接的网络通信(workerGroup)。 2. 创建了一个线程池,使用`ThreadPoolExecutor`来执行异步任务。线程池的大小与要绑定的端口数量一致。 3. 创建了一个`ServerBootstrap`对象,并将bossGroup和workerGroup设置到该对象中。 4. 调用`bootstrap.channel(NioServerSocketChannel.class)`设置服务器的通道类型为NIO类型。 5. 使用`option()`方法设置服务器的一些配置选项。例如,使用`ChannelOption.SO_BACKLOG`设置连接请求的最大队列长度为128,使用`ChannelOption.SO_KEEPALIVE`设置保持连接状态为true。 6. 使用`LoggingHandler`设置日志级别为INFO。 7. 调用`consumer.accept(bootstrap)`方法,允许用户自定义配置`ServerBootstrap`实例。 8. 使用for循环遍历输入的`ports`数组。对于每个端口,如果不是第一个端口(i > 0),则使用`CompletableFuture.runAsync()`方法在线程池中异步执行绑定操作,并等待绑定完成。 9. 在捕获异常的块中,记录绑定异常并打印日志。 10. 在循环结束后,再次使用`bootstrap.bind(ports[0])`方法绑定第一个端口,并等待绑定完成。 11. 在finally块中,调用`service.shutdown()`来关闭线程池,释放资源。然后,调用`workerGroup.shutdownGracefully()`和`bossGroup.shutdownGracefully()`来优雅地关闭线程组,释放资源。 问题检查: - 通过使用线程池来异步执行端口绑定和关闭操作,可以控制并发度,避免创建过多的线程。这样可以更好地管理系统资源。 - 添加了一些服务器的配置选项,如`SO_BACKLOG`和`SO_KEEPALIVE`,以提供更好的性能和可靠性。 综上所述,这段代码在原有的异步绑定和关闭操作的基础上,进一步添加了线程池的支持和服务器的配置选项,提高了代码的可扩展性、性能和可靠性。同时,在finally块中关闭了线程池和释放了资源。
评论 8
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值