(二)Redis的事件

2019-04-11  本文已影响0人  纸中圆

  Redis服务器是一个事件驱动程序,服务器需要处理以下两类事件:

文件事件

  Redis基于Reactor模式开发了自己的网络事件处理器--文件事件处理器:

文件事件处理器的构成

  文件事件处理器由套接字、I/O多路复用程序、文件事件分派器和事件处理器构成,如下图:

文件事件处理器组成

  文件事件是对套接字操作的抽象,每当一个套接字准备好执行连接应答、写入、读取、关闭等操作时,就会产生一个文件事件。因为一个服务器会连接多个套接字,所以多个文件事件可能会并发地出现。
  I/O多路复用程序负责监听多个套接字,并向文件事件分派器传送那些产生了事件的套接字。
  尽管多个文件事件可能会并发地出现,但I/O多路复用程序总是会将所有产生事件的套接字都放到一个队列里面,然后通过这个队列,以有序(sequentially)、同步(synchronously)、每次一个套接字的方式向文件事件分派器传送套接字。当一个套接字产生的事件被处理完毕后(该套接字为事件所关联的事件处理器执行完毕),I/O多路复用程序才会继续向文件事件分派传送下一个套接字,如下图:

I/O多路复用程序通过队列向文件事件分派器传送套接字
  文件事件分派器接收到I/O多路复用程序传来的套接字,并根据套接字产生的事件类型调用相应的事件处理器。
  服务器会为执行不同任务的套接字关联不同的事件处理器,这些处理器是一个个函数,它们定义了某个事件发生时,服务器应该执行的动作

I/O多路复用程序的实现

  Redis的I/O多路复用程序的所有功能都是通过包装常见的select、epoll、evport、kqueue这些I/O多路复用函数库实现的。因为Redis为每个函数库都实现了相同的API,所以I/O多路复用程序的底层实现是可以互换的(Redis定义相应的规则所以在编译时自动选择性能最高的库来作为其底层实现),如下图:

Redis中的I/O多路复用

事件的类型

  I/O多路复用程序可以监听多个套接字的ae.h/AE_READABLE事件和ae.h/AE_WRITABLE事件,这连累事件和套接字操作之间的关系如下:

文件事件的处理器

  Redis为文件事件编写了多个处理器,分别用于实现不同的网络通信需求,下面简单介绍:

时间事件

  Redis的时间事件分为下面2类:

时间事件实现

  服务器将所有时间事件都放在一个无序链表中,当时间事件处理器被执行时, 它遍历所有链表中的时间事件, 检查它们的到达事件(when 属性), 并执行其中的已到达事件

注:正常模式下的Redis服务器只使用serverCron一个时间事件。

时间事件应用实例:serverCron函数

  持续运行的Redis服务器需要定期对自身资源和状态进行检查和调整,从而确保服务器可以长期、稳定地运行,这些定期操作由redis.c/serverCrom函数负责执行,它的主要工作包括:

事件的调度与执行

  因为服务器中同时存在文件事件和时间事件,所以服务器必须对这两种事件进行调度,决定何时应该处理文件事件,何时处理时间事件,以及花多少时间来处理它们等等。
  文件事件和时间事件之间是合作关系,服务器会轮流处理这两种事件,并且处理过程中也不会进行互相抢占,但因此实际处理时间事件的事件比预定到达的时间会晚一些。

参考资料

《redis设计与实现》(第二版)

上一篇 下一篇

猜你喜欢

热点阅读