Netty客户端channel主动关闭时,服务端感知并关闭channel的过程

https://blog.csdn.net/u010013573/article/details/86485856

  1. 关闭channel时,会调用unsafe的close,故可以在服务端中,在AbstractChannel#AbstractUnsafe的close(final ChannelPromise promise)方法上设置断点:这个是比较顶层的方法。
    在这里插入图片描述

  2. 往下看底层开始时的调用栈:在selector接收到该channel的读写事件时,调用processSelectedKey处理进行处理,如下可知,
    此时产生的是一个OP_READ事件,即客户端主动关闭时,向服务端写了数据过来,服务端调用unsafe.read(),读取这个数据。
    其中unsafe为NioSocketChannel#NioSocketChannelUnsafe。
    在这里插入图片描述

  3. 继续debug到unsafe.read里面,发现读到的数据为空,故close = localReadCount < 0,返回为true。
    read方法在AbstractNioByteChannel#NioByteUnsafe中定义。
    在这里插入图片描述如下localReadAmount为-1,
    在这里插入图片描述
    而根据ByteBuf接口的setBytes的定义可知,对方(这里是客户端)channel关闭时,返回-1
    在这里插入图片描述
    close为true,则调用closeOnRead(pipeline),关闭这个channel。
    在这里插入图片描述

  4. closeOnRead的方法执行如下:allowHalfClosure=false,故调用的是close(voidPromise)。
    在这里插入图片描述

  5. close(voidPromise)的执行如下:调用doClose0关闭底层的socket,调用fireChannelInactiveAndDeregister方法,触发pipeline中,从head开始,往下传给
    InboundHandler,并执行channelInactive方法。其中close方法在AbstractChannel#AbstractUnsafe中定义。
    在这里插入图片描述
    在这里插入图片描述
    在这里插入图片描述
    在这里插入图片描述

总结

  • 客户端channel主动关闭连接时,会向服务端发送一个写请求,然后服务端channel所在的selector会监听到一个OP_READ事件,然后
    执行数据读取操作,而读取时发现客户端channel已经关闭了,则读取数据字节个数返回-1,然后执行close操作,关闭该channel对应的底层socket,
    并在pipeline中,从head开始,往下将InboundHandler,并触发handler的channelInactive和channelUnregistered方法的执行,以及移除pipeline中的handlers一系列操作。

  • 推荐阅读:Netty网络处理之客户端主动关闭

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Netty 中,客户端服务端发送数据需要经过以下步骤: 1. 创建一个 Bootstrap 实例,用于启动客户端。 2. 设置客户端的 EventLoopGroup,用于处理网络事件。 3. 配置客户端Channel,包括 Channel 的类型、ChannelPipeline 的配置等。 4. 连接到服务端,使用 connect() 方法连接到服务端。 5. 在连接成功后,向服务端发送数据,可以通过 ChannelHandlerContext 的 writeAndFlush() 方法实现。 以下是一个简单的示例代码: ```java EventLoopGroup group = new NioEventLoopGroup(); Bootstrap bootstrap = new Bootstrap(); bootstrap.group(group) .channel(NioSocketChannel.class) .handler(new ChannelInitializer<SocketChannel>() { @Override protected void initChannel(SocketChannel ch) throws Exception { ChannelPipeline pipeline = ch.pipeline(); pipeline.addLast(new StringEncoder()); pipeline.addLast(new StringDecoder()); pipeline.addLast(new ClientHandler()); } }); ChannelFuture future = bootstrap.connect("localhost", 8888).sync(); future.channel().writeAndFlush("Hello, server!"); future.channel().closeFuture().sync(); group.shutdownGracefully(); ``` 在上面的示例中,我们创建了一个 Bootstrap 实例,并设置了客户端的 EventLoopGroup 和 Channel。然后使用 connect() 方法连接到服务端,并在连接成功后向服务端发送了一条消息。注意,在这里我们使用了 ChannelHandlerContext 的 writeAndFlush() 方法来发送数据。最后,我们等待连接关闭关闭 EventLoopGroup。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值