Channel提供了3个方法用来实现关闭清理功能:disconnect,close,deregister。本章重点分析这个3个方法的功能的NIO实现。
disconnect实现: 断开连接
disconnect方法的调用栈如下:
1 io.netty.channel.AbstractChannel#disconnect()2 io.netty.channel.DefaultChannelPipeline#disconnect()3 io.netty.channel.AbstractChannelHandlerContext#disconnect()4 io.netty.channel.AbstractChannelHandlerContext#disconnect(io.netty.channel.ChannelPromise)5 io.netty.channel.AbstractChannelHandlerContext#invokeDisconnect6 io.netty.channel.DefaultChannelPipeline.HeadContext#disconnect7 io.netty.channel.AbstractChannel.AbstractUnsafe#disconnect8 io.netty.channel.socket.nio.NioSocketChannel#doDisconnect9 io.netty.channel.socket.nio.NioSocketChannel#doClose
disconnect稍微复杂一些, 在io.netty.channel.AbstractChannelHandlerContext#disconnect(io.netty.channel.ChannelPromise)实现中,会根据channel是否支持disconnect操作来决定下一步动作:if (!channel().metadata().hasDisconnect()) {
next.invokeClose(promise);
} else {
next.invokeDisconnect(promise);
}
之所以这样设计,是因为TCP和UDP的disconnect含义是不一样的,对TCP来说disconnect就是关闭socket;对UDP来说,它没有连接的概念,默认情况下通过udp socket发送数据需要指定远程地址,但如果调用connect之后,就不需指定这个地址,数据报会被发送到connect指定的地址上,disconnect含义是删除connect指定的地址,发送数据时必须指定地址。所以在NIO的Channel实现中,TCP的disconnect是调用socket的close方法,UDP的disconnect是调用socket的disconnect方法,下面是两种不同的disconnect实现。
//TCP io.netty.channel.socket.nio.NioSocketChannel#doDisconnect@Overrideprotected void doDisconnect() throws Exception {
doClose();
}
@Overrideprotected void doClose() throws Exception { super.doClose();
javaChannel().close();
}//UDP io.netty.channel.socket.nio.NioDatagramChannel#doDisconnect@Overrideprotected void doDisconnect() throws Exception {
javaChannel().disconnect();
}
io.netty.channel.AbstractChannel.AbstractUnsafe#disconnect实现了disconnect的逻辑,先调用doDisconnect方法,这个方法是io.netty