最近在看 UNIX 网络编程并研究了一下 redis 的实现,感觉 Redis 的源代码十分适合阅读和分析,其中 I/O 多路复用(mutiplexing)部分的实现非常干净和优雅,在这里想对这部分的内容进行简单的整理 。
文章插图
几种 I/O 模型
为什么 Redis 中要使用 I/O 多路复用这种技术呢?首先,Redis 是跑在单线程中的,所有的操作都是按照顺序线性执行的 。
但是由于读写操作等待用户输入或输出都是阻塞的,所以 I/O 操作在一般情况下往往不能直接返回 。
这会导致某一文件的 I/O 阻塞导致整个进程无法对其他客户提供服务,而 I/O 多路复用就是为了解决这个问题而出现的 。
Blocking I/O
先来看一下传统的阻塞 I/O 模型到底是如何工作的:当使用 Read 或者 Write 对某一个文件描述符(File Descriptor 以下简称 FD)进行读写时 。
如果当前 FD 不可读或不可写,整个 Redis 服务就不会对其他的操作作出响应,导致整个服务不可用 。
这也就是传统意义上的,我们在编程中使用最多的阻塞模型:
文章插图
阻塞模型虽然开发中非常常见也非常易于理解,但是由于它会影响其他 FD 对应的服务,所以在需要处理多个客户端任务的时候,往往都不会使用阻塞模型 。
I/O 多路复用
虽然还有很多其他的 I/O 模型,但是在这里都不会具体介绍 。阻塞式的 I/O 模型并不能满足这里的需求,我们需要一种效率更高的 I/O 模型来支撑 Redis 的多个客户(redis-cli) 。
这里涉及的就是 I/O 多路复用模型了:
文章插图
在 I/O 多路复用模型中,最重要的函数调用就是 select,该方法的能够同时监控多个文件描述符的可读可写情况,当其中的某些文件描述符可读或者可写时,select 方法就会返回可读以及可写的文件描述符个数 。
关于 select 的具体使用方法,在网络上资料很多,这里就不过多展开介绍了;
与此同时也有其它的 I/O 多路复用函数 epoll/kqueue/evport,它们相比 select 性能更优秀,同时也能支撑更多的服务 。
Reactor 设计模式
Redis 服务采用 Reactor 的方式来实现文件事件处理器(每一个网络连接其实都对应一个文件描述符)
文章插图
文件事件处理器使用 I/O 多路复用模块同时监听多个 FD,当 accept、read、write 和 close 文件事件产生时,文件事件处理器就会回调 FD 绑定的事件处理器 。
虽然整个文件事件处理器是在单线程上运行的,但是通过 I/O 多路复用模块的引入,实现了同时对多个 FD 读写的监控,提高了网络通信模型的性能,同时也可以保证整个 Redis 服务实现的简单 。
I/O 多路复用模块
I/O 多路复用模块封装了底层的 select、epoll、avport 以及 kqueue 这些 I/O 多路复用函数,为上层提供了相同的接口 。
文章插图
在这里我们简单介绍 Redis 是如何包装 select 和 epoll 的,简要了解该模块的功能,整个 I/O 多路复用模块抹平了不同平台上 I/O 多路复用函数的差异性,提供了相同的接口:
static int aeApiCreate(aeEventLoop *eventLoop) static int aeApiResize(aeEventLoop *eventLoop, int setsize) static void aeApiFree(aeEventLoop *eventLoop) static int aeApiAddEvent(aeEventLoop *eventLoop, int fd, int mask) static void aeApiDelEvent(aeEventLoop *eventLoop, int fd, int mask) static int aeApiPoll(aeEventLoop *eventLoop, struct timeval *tvp) 同时,因为各个函数所需要的参数不同,我们在每一个子模块内部通过一个 aeApiState 来存储需要的上下文信息:
// select typedef struct aeApiState {fd_set rfds, wfds;fd_set _rfds, _wfds; } aeApiState;// epoll typedef struct aeApiState {int epfd;struct epoll_event *events; } aeApiState; 这些上下文信息会存储在 eventLoop 的 void *state 中,不会暴露到上层,只在当前子模块中使用 。
封装 Select 函数
Select 可以监控 FD 的可读、可写以及出现错误的情况 。在介绍 I/O 多路复用模块如何对 Select 函数封装之前,先来看一下 Select 函数使用的大致流程:
- 初始化一个可读的 fd_set 集合,保存需要监控可读性的 FD 。
推荐阅读
- 简单又实用的台湾日月潭红茶冲泡方法
- 实现扫码登陆的最简单方案与原理
- 台湾日月潭红茶的冲泡方法 简单又实用
- 伞竹怎么养才能长的有粗又高的简单介绍
- 茶化石那些传闻 为什么不建议喝碎银子?
- 永远不要在春节的时候相亲 大龄单身过年该不该相亲
- 一亿像素影像横评 hm2是什么单位
- 为什么现在的男生都喜欢姐姐,为什么现在的00后喜欢姐弟恋
- 如何选购坦洋工夫红茶坦洋工夫为什么这么火
- 事业单位的薪级工资是怎么计算的?