IO线程模型Reactor模型

本文围绕Java网络IO线程模型展开,先介绍传统阻塞I/O服务模型的特点与问题,随后重点阐述Reactor模型,包括单Reactor单线程、单Reactor多线程、主从Reactor多线程三种类型,分析各自优缺点及使用场景,最后总结Reactor模型响应快、扩展性和复用性好的优点。
概述

前言:
要对IO模型、多路复用、java NIO网络编程有一定的理解才能看懂。

不同的线程模型,对程序的性能影响很大,要搞钱Netty的线程模型,就要了解一下各种线程模型。

目前存在的线程模型有:

  1. 传统阻塞I/O服务模型。
  2. Reactor模型。
首先来说说传统的网络IO线程模型:

在这里插入图片描述
特点:

  1. 采用阻塞IO模式获取输入的数据。
  2. 每个连接都需要独立的线程完成数据的输入(会造成阻塞)、业务处理、数据返回(会造成阻塞)。也就是一个线程就负责一个客户端连接的所有东西。

问题:

  1. 当并发量很大时,需要创建大量线程,会耗尽系统资源,就算使用线程池,也会造成大量连接被阻塞。
  2. 这些线程会被阻塞在read、write操作上面,造成资源浪费。
Reactor模型

Reactor是解决针对传统IO阻塞模式的缺点的解决方法。

Reactor模式就是使用一个线程,然后配合非阻塞IO,IO多路复用模式来代替传统的IO阻塞模式。

Reactor模式对应的叫法有:反应器模式、分发者模式、通知者模式。

根据Reactor数量和处理资源线程池线程数量的不同,可以将Reactor分为三种典型的类型:

  1. 单Reactor单线程。
  2. 单Reactor多线程。
  3. 主从Reactor多线程。

Reactor模型的角色:

  • Reactor:用于处理客户端连接的组件,通常一个Reactor
### Reactor 线程模型的工作原理 Reactor 模式是一种用于处理输入/输出操作的事件驱动架构,特别适用于高并发网络编程环境。在这种模式下,应用程序通过一个或多个线程来监听并响应各种 I/O 事件。 #### 单 Reactor线程模型 最简单的形式是单 Reactor线程模型,在此模型中只有一个线程负责所有的读写操作以及连接请求的接收。当有新的连接到来时,这个唯一的线程会先完成对该连接的初始化设置;之后每当发生任何可读写的 socket 就绪状态变化时,也是由这同一个线程来进行相应的业务逻辑处理[^2]。 然而,由于所有任务都在同一根线上串行执行,因此如果某个耗时较长的任务占据了 CPU 时间片,则其他等待中的任务会被延迟处理,从而影响整体性能表现。 #### 单 Reactor线程模型 为了克服上述局限性,可以采用单 Reactor 多线程方案。这里仍然存在单一入口点用来侦听新到达的数据包或者建立的新链接,但是针对每一个发生的 IO 事件(比如数据准备就绪),都会创建一个新的工作线程去具体承担实际的数据传输职责。这种方式能够在一定程度上缓解因个别长时间运行的操作而导致整个系统变得缓慢的问题[^1]。 不过需要注意的是,频繁地创建销毁临时性的 worker threads 可能带来额外开销,并且可能导致资源竞争现象的发生。 #### 主从 Reactor线程模型 更高级别的设计思路体现在所谓的 "main-sub reactor" 或者说 “master-slave reactors”。在这个版本里面引入了两层结构: - **主 Reactor (Master)**:专门从事于监听端口上的新连接尝试,并将其转发给预先设定好的一组子处理器之一; - **从 Reactors (Slaves)** :各自独立运作,专注于管理自己所辖范围内的活动套接字对象及其关联的各种异步通知机制。一旦发现目标文件描述符处于活跃态即刻唤醒对应的 handler 进入下一步骤——调用 read/write 函数实施真正的交互动作[^3]。 这样的安排不仅能够有效减轻主线程负担,而且还可以充分利用现代计算机硬件所提供的多核特性,进而达到更高的吞吐量指标。 ### 实际应用场景下的实现方式 考虑到不同框架对于内部组件的具体定义有所差异,下面给出基于 Java 平台之上构建的一个简化版 netty server demo 来展示如何运用 master/slave 架构搭建高性能的服务端程序[^4]。 ```java import io.netty.bootstrap.ServerBootstrap; import io.netty.channel.ChannelFuture; import io.netty.channel.EventLoopGroup; import io.netty.channel.nio.NioEventLoopGroup; import io.netty.channel.socket.nio.NioServerSocketChannel; public class NettyServer { public static void main(String[] args) throws Exception { EventLoopGroup bossGroup = new NioEventLoopGroup(); // Master thread group for accepting connections. EventLoopGroup workerGroup = new NioEventLoopGroup(); // Slave thread groups for handling actual data transfer. try { ServerBootstrap b = new ServerBootstrap(); b.group(bossGroup, workerGroup) .channel(NioServerSocketChannel.class); ChannelFuture f = b.bind(8080).sync(); System.out.println("Server started and listening on port 8080"); f.channel().closeFuture().sync(); } finally { bossGroup.shutdownGracefully(); workerGroup.shutdownGracefully(); } } } ``` 这段代码片段展示了如何使用 Netty 库快速建立起具备良好扩展性稳定性的 TCP/IP 服务器实例。其中 `bossGroup` `workerGroup` 分别对应着前面提到过的两种类型的反应堆实体,它们共同协作实现了高效稳定的通信链路维护功能[^5]。
评论 2
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值