Redis(设计与实现):62---发布与订阅之频道的订阅与退订(SUBSCRIBE命令、UNSUBCRIBE命令、pubsub_channels属性)

pubsub_channels字典

  • Redis将所有频道的订阅关系都保存在服务器状态的pubsub_channels字典里面,这个字典的键是某个被订阅的频道,而键的值则是一个链表,链表里面记录了所有订阅这个频道的客户端:
struct redisServer {
    // ...
    //保存所有频道的订阅关系
    dict *pubsub_channels;
    // ...
};
  • 比如说,下图就展示了一个pubsub_channels字典示例,这个字典记录了以下信息:
    • client-1、client-2、client-3三个客户端正在订阅"news.it"频道
    • 客户端client-4正在订阅"news.sport"频道
    • client-5和client-6两个客户端正在订阅"news.business"频道

Redis(设计与实现):62---发布与订阅之频道的订阅与退订(SUBSCRIBE命令、UNSUBCRIBE命令、pubsub_channels属性)_第1张图片

一、何为频道的订阅

  • 通过执行SUBSCRIBE命令,客户端可以订阅一个或多个频道,从而成为这些频道的订阅者(subscriber):每当有其他客户端使用PUBLISH命令向被订阅的频道发送消息(message)时,频道的所有订阅者都会收到这条消息

演示案例

  • 假设A、B、C三个客户端都执行了命令,那么这三个客户端就是"news.it"频道的订阅者,如下图所示:
SUBSCRIBE "news.it"

Redis(设计与实现):62---发布与订阅之频道的订阅与退订(SUBSCRIBE命令、UNSUBCRIBE命令、pubsub_channels属性)_第2张图片

  • 如果这时某个客户端执行下面命令,向"news.it"频道发送消息"hello",那么"news.it"的三个订阅者都将收到这条消息,如下图所示:
PUBLISH "news.it" "hello"

二、频道的订阅(SUBSCRIBE命令)

  • 每当客户端执行SUBSCRIBE命令订阅某个或某些频道的时候,服务器都会将客户端与被订阅的频道在pubsub_channels字典中进行关联
  • 根据频道是否已经有其他订阅者,关联操作分为两种情况执行:
    • 如果频道已经有其他订阅者,那么它在pubsub_channels字典中必然有相应的订阅者链 表,程序唯一要做的就是将客户端添加到订阅者链表的末尾
    • 如果频道还未有任何订阅者,那么它必然不存在于pubsub_channels字典,程序首先要在 pubsub_channels字典中为频道创建一个键,并将这个键的值设置为空链表,然后再将客户端 添加到链表,成为链表的第一个元素
  • 举个例子,假设服务器pubsub_channels字典的当前状态如上图所示,那么当客户端 client-10086执行命令之后,pubsub_channels字典将更新至下图所示的状态,其中用虚线包围的是新添加的节点:
SUBSCRIBE "news.sport" "news.movie"

Redis(设计与实现):62---发布与订阅之频道的订阅与退订(SUBSCRIBE命令、UNSUBCRIBE命令、pubsub_channels属性)_第3张图片

  • SUBSCRIBE命令的实现可以用以下伪代码来描述:
def subscribe(*all_input_channels):
    # 遍历输入的所有频道
    for channel in all_input_channels:
        # 如果channel 不存在于pubsub_channels 字典(没有任何订阅者)
        # 那么在字典中添加channel 键,并设置它的值为空链表
        if channel not in server.pubsub_channels:
            server.pubsub_channels[channel] = []
        # 将订阅者添加到频道所对应的链表的末尾
        server.pubsub_channels[channel].append(client)

三、频道的退订(UNSUBSCRIBE命令)

  • UNSUBSCRIBE命令的行为和SUBSCRIBE命令的行为正好相反,当一个客户端退订某个或某些频道的时候,服务器将从pubsub_channels中解除客户端与被退订频道之间的关联:
    • 程序会根据被退订频道的名字,在pubsub_channels字典中找到频道对应的订阅者链表, 然后从订阅者链表中删除退订客户端的信息
    • 如果删除退订客户端之后,频道的订阅者链表变成了空链表,那么说明这个频道已经没有任何订阅者了,程序将从pubsub_channels字典中删除频道对应的键
  • 举个例子,假设pubsub_channels的当前状态如下面第一章图所示,那么当客户端client-10086执 行命令之后,图中用虚线包围的两个节点将被删除:
    • 在pubsub_channels字典更新之后,client-10086的信息已经从"news.sport"频道 和"news.movie"频道的订阅者链表中被删除了。
    • 另外,因为删除client-10086之后,频道"news.movie"已经没有任何订阅者,因此 键"news.movie"也从字典中被删除了
UNSUBSCRIBE "news.sport" "news.movie"

Redis(设计与实现):62---发布与订阅之频道的订阅与退订(SUBSCRIBE命令、UNSUBCRIBE命令、pubsub_channels属性)_第4张图片Redis(设计与实现):62---发布与订阅之频道的订阅与退订(SUBSCRIBE命令、UNSUBCRIBE命令、pubsub_channels属性)_第5张图片

  • UNSUBSCRIBE命令的实现可以用以下伪代码来描述:
def unsubscribe(*all_input_channels):
    #遍历要退订的所有频道
    for channel in all_input_channels:
        # 在订阅者链表中删除退订的客户端
        server.pubsub_channels[channel].remove(client)

        # 如果频道已经没有任何订阅者了(订阅者链表为空)
        # 那么将频道从字典中删除
        if(len(server.pubsub_channels[channel])==0:
            server.pubsub_channels.remove(channel)

 

你可能感兴趣的:(Redis(设计与实现))