《Reids 设计与实现》第二十章 监视器
一、简介
通过执行 MONITOR 命令,客户端可以将自己变为一个监视器,实时地接收并打印处服务器当前处理地命令请求的相关信息:
每当一个客户端向服务器发送一条命令请求时,服务器除了会处理这条命令请求之外,还会将关于这条命令请求的信息发送给所有监视器,如图 23-1 所示
二、成为监视器
发送 MONITOR 命令可以让一个普通客户端变为一个监视器,该命令的实现原理可以用以下伪代码来实现:
def MONITOR():
#打开客户端的监视器标志
client.flags |= REDIS_MONITOR
#将客户端添加到服务器状态的 monitors 链表的末尾
server.monitors.append(client)
#向客户端返回 OK
send_reply("OK")
举个例子,如果客户端 c10086 向服务器发送 MONITOR 命令,那么这个客户端的 REDIS_MONITOR 标志会被打开,并且这个客户端本身会被添加到 monitors 链表的表尾
假设客户端 c10086 发送 MONITOR 命令之前,monitors 链表的状态如 24-2 所示,那么在服务器执行客户端 c10086 发送的 MONITOR 命令之后,monitors 链表将被更新为图 24-3 所示的状态
三、向监视器发送命令信息
服务器在每次处理命令请求之前,都会调用 replicationFeedMonitors 函数,由这个函数将被处理的命令请求的相关信息发送给各个监视器
以下是 replicationFeedMonitors 函数的伪代码定义,函数首先根据传入的参数创建信息,然后将信息发送给所有监视器:
def replicationFeedMonitors(client, monitors, dbid, argv, argc):
#根据执行命令的客户端、当前数据库的号码、命令参数、命令参数个数等参数
#创建要发送给各个监视器的信息
msg = create_message(client, dbid, argv, argc)
#遍历所有监视器
for monitor in monitors:
#将信息发送给监视器
send message(monitor, msg)
举个例子,假设服务器在时间 1378822257.329412,根据 IP 为 127.0.0.1、端口号为 56604 的客户端发送的命令请求,对 0 号数据库执行命令 KEYS*,那么服务器将创建以下信息:
如果服务器 monitors 链表的当前状态如图 24-3 所示,那么服务器会分别将信息发送给 c128、c256、c512 和 c10086 四个监视器,如图 24-4 所示
四、重点回顾
- 客户端可以通过执行 MONITOR 命令,将客户端转换成监视器,接收并打印服务器处理的每个命令请求的相关信息
- 当一个客户端从普通客户端变为监视器时,该客户端的 REDIS_MONITOR 标识会被打开
- 服务区将所有监视器都记录在 monitors 链表中
- 每次处理命令请求时,服务区都会遍历 monitors 链表,将相关信息发送给监视器