发布日期:2025-05-23 10:49 点击次数:179
Netty 的线程模型是基于主从多 Reactor 模型。 Netty 中网络的连接事件 (OP_ACCEPT) 由 Main Reactor 线程组实现,即 Boss Group,通常只需设置一个线程。
网络的读写操作由 Work Group (Sub Reactor) 线程组来实现,线程的个数默认为 2 * CPU Core,一个 Channel 绑定到其中一个 Work 线程,一个 Work 线程中可以绑定多个 Channel。
在 Netty 中编码、解码等操作会被封装成一个一个事件处理器 (ChannelHandler),那这些 Handler 是在 IO 线程池中执行?
默认情况下 ChannelHandler 是在 IO 线程中执行,那如何改变默认行为呢?其关键代码如下: 关键点:在将事件处理器添加到事件链时可以指定在哪个线程池中执行,如果不指定则为 IO 线程中执行。
面试官:通常业务操作会专门开辟一个线程池,那业务处理完成之后,如何将响应结果通过 IO 线程写入到网卡中呢? 业务线程调用 Channel 对象的 write 方法并不会立即写入网络,只是将数据放入一个待写入队列 (缓存区),然后 IO 线程每次执行事件选择后,会从待写入缓存区中获取写入任务,将数据真正写入到网络中,数据到达网卡之前会经过一系列的 Channel Handler (Netty 事件传播机制),最终写入网卡。
最后再来介绍一下 Netty 中 IO 线程的大体工作流程。 IO 线程处理的关键点:
每一 IO 线程在执行上述操作时是串行执行的,即注册在一个 Selector (事件选择器) 中的所有通道,** 同一时间只有一个通道的事件被处理。** 这也是为什么 NIO 应对大文件传输时不具备优势的根本原因。
IO 线程在处理完所有就绪事件后,还会从任务队列 (Task Queue) 获取任务,例如上文中提到的业务线程在执行完业务后需要将返回结果写入网络,Netty 中所有的网络读写操作只能在 IO 线程中真正获得运行,故业务线程需要将带写入的响应结果封装成 Task,放入到 IO 线程任务队列中。