【笔记】《Redis设计与实现》chapter12 事件
12.1 文件事件
Redis基于Reactor模式开发了自己的网络事件处理器:这个处理器被称为文件时间处理器:
- 文件时间处理器使用IO多路复用程序来同时监听多个套接字,并根据套接字目前执行的任务来为套接字关联不用的事件处理器
- 当被监听的套接字准备好执行连接应答(accept)、读取(read)、写入(write)、关闭(close)等操作时,与操作相对应的文件时间就会产生,这是文件事件处理器就会调用套接字之前关联好的事件处理器来处理这些事件
文件事件处理器的构成
IO多路复用程序将同时响应的套接字放在一个队列中,每次向文件事件分派器分派一个套接字
IO多路复用程序的实现
事件的类型
- 当套接字可读时,产生AE_READABLE事件
- 当套接字可写时,产生AE_WRITEABLE事件
- 当同一个套接字,可读可写时,先处理AE_READABLE事件,再处理AE_WRITEABLE事件
API
文件事件的处理器
12.2 时间事件
Redis 的事件事件分为以下两类:
- 定时事件:让一段程序在指定的时间之后执行一次。
- 周期性事件:让一段程序每隔指定时间就执行一次。
一个时间事件主要由以下三个属性组成:
- id:服务器为时间事件创建的全局唯一ID
- when:毫秒精度的UNIX时间戳
- timeProc:时间事件处理器,一个函数
实现
服务器将所有时间事件都放在一个无序链表中,每当时间事件执行器运行时,它就遍历整个链表,查找所有已到达的时间事件
时间事件应用实例:serverCron函数
它的主要工作包括:
- 更新服务器的各类统计信息
- 清理数据库中的过期键值对
- 关闭和清理连接失效的客户端
- 尝试进行AOF 或 RDB持久化操作
- 如果服务器是主服务器
- 如果处于集群模式,对集群进行定期同步和连接测试
12.3 事件的调度与执行
- aeApiPoll函数的最大阻塞时间由到达时间最接近当前时间的时间事件决定,比秒服务器对时间事件进行频繁的轮询,也避免阻塞时间过长
- 因为文件时间是随机出现的,如果等待并处理完一次文件事件之后,仍未有任何时间事件到达,那么服务器继续等待,逐渐逼近时间事件
- 对文件事件和时间事件的处理都是同步、有序、原子地执行的