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方法是等待异步操作执行完毕.