Redis发布订阅
在现代的软件开发中,数据存储和管理是至关重要的一环。Redis,作为一个开源的、内存中的数据结构存储系统,以其出色的性能和灵活的数据结构,赢得了开发者们的广泛喜爱。它不仅可以用作数据库,还可以用作缓存和消息代理。今天,我们要探讨的是 Redis 中一个强大的功能——发布订阅模式。
发布订阅模式是一种消息通信模式,发送者(发布者)发送消息,订阅者接收消息。在 Redis 中,客户端可以订阅任意数量的频道,当有新消息通过
PUBLISH
命令发送给频道时,这个消息会被发送给订阅它的所有客户端。在接下来的文章中,我们将详细介绍 Redis 的发布订阅模式,包括它的工作原理,如何使用,以及一些常见的使用场景。无论你是刚接触 Redis,还是已经有一定的使用经验,我相信你都能从这篇文章中学到一些新的知识。
文章目录
- @[toc]
- 1、Redis发布订阅介绍
- 1.1、Redis发布订阅概述
- 1.2、Redis发布订阅与消息队列的区别
- 2、Redis发布订阅的原理
- 2.1、Redis实现发布订阅的原理
- 2.2、Redis实现发布订阅的底层结构
- 3、发布订阅的命令
- 3.1、SUBSCRIBE命令
- 3.2、UNSUBSCRIBE命令
- 3.3、PUBLISH命令
文章目录
- @[toc]
- 1、Redis发布订阅介绍
- 1.1、Redis发布订阅概述
- 1.2、Redis发布订阅与消息队列的区别
- 2、Redis发布订阅的原理
- 2.1、Redis实现发布订阅的原理
- 2.2、Redis实现发布订阅的底层结构
- 3、发布订阅的命令
- 3.1、SUBSCRIBE命令
- 3.2、UNSUBSCRIBE命令
- 3.3、PUBLISH命令
1、Redis发布订阅介绍
1.1、Redis发布订阅概述
Redis 的发布订阅(Pub/Sub)是一种消息通信模式:发送者(pub)发送消息,订阅者(sub)接收消息。Redis 客户端可以订阅任意数量的频道。当有新消息通过 PUBLISH
命令发送给频道时,这个消息会被发送给订阅它的所有客户端
1.2、Redis发布订阅与消息队列的区别
Redis的发布订阅(Pub/Sub)和消息队列是两种不同的消息传递模式,它们的主要区别在于消息的处理方式和使用场景。
消息的处理方式:
- 在 Redis 的发布订阅模式中,消息是即时的,也就是说,当消息发布后,只有当前在线且订阅了该频道的客户端才能收到这个消息,消息不会被存储,一旦发布,当前没有在线的客户端将无法接收到这个消息。
- 在消息队列中,消息是持久化的,消息被发送到队列后,会一直在队列中等待被消费,即使没有在线的消费者,消息也不会丢失,消费者下次上线后可以继续从队列中获取到消息。
使用场景:
- Redis 的发布订阅模式通常用于实现实时消息系统,比如实时聊天、实时推送通知等。
- 消息队列通常用于异步处理,解耦复杂系统,比如电商系统中的下单、支付、库存处理等操作,通过消息队列可以使这些操作异步处理,提高系统的响应速度。
总的来说,Redis 的发布订阅模式更适合实时、必须立即处理的场景,而消息队列更适合异步处理、耗时操作的场景。
2、Redis发布订阅的原理
2.1、Redis实现发布订阅的原理
Redis 的发布订阅(Pub/Sub)模式的原理主要涉及到三个命令:SUBSCRIBE
,UNSUBSCRIBE
和 PUBLISH
。
-
SUBSCRIBE
命令:当客户端发送SUBSCRIBE
命令订阅一个或多个频道时,Redis 服务器会将这些频道添加到该客户端的订阅频道列表中,并向客户端返回一个确认订阅的消息。 -
UNSUBSCRIBE
命令:当客户端发送UNSUBSCRIBE
命令退订一个或多个频道时,Redis 服务器会将这些频道从客户端的订阅频道列表中移除,并向客户端返回一个确认退订的消息。 -
PUBLISH
命令:当客户端发送PUBLISH
命令发布消息到某个频道时,Redis 服务器会查找所有订阅了这个频道的客户端,并将消息发送给这些客户端。
在 Redis 的内部实现中,服务器维护了一个字典,字典的键是频道的名字,字典的值是一个链表,链表中存储了所有订阅了这个频道的客户端。当有新消息发布到某个频道时,服务器只需要查找这个字典,就可以快速找到需要接收这个消息的所有客户端。
这种实现方式使得 Redis 的发布订阅模式具有很高的效率,可以支持大量的客户端同时订阅同一个频道。
2.2、Redis实现发布订阅的底层结构
Redis 的发布订阅(Pub/Sub)模式的底层结构主要包括两个部分:客户端结构和服务器的Pub/Sub结构。
客户端结构:每个 Redis 客户端都有一个 pubsub_channels
和 pubsub_patterns
两个属性,分别用于存储该客户端订阅的频道和模式。
-
pubsub_channels
:这是一个字典,键是订阅的频道名,值是NULL。当客户端订阅一个新的频道时,频道名会被添加到这个字典中;当客户端退订一个频道时,频道名会从这个字典中删除。 -
pubsub_patterns
:这是一个链表,存储了所有订阅的模式。每个模式都是一个redisPubSubPattern
结构,包含了模式本身和订阅这个模式的客户端。
服务器的Pub/Sub结构:Redis 服务器维护了一个 pubsub_channels
字典和一个 pubsub_patterns
链表,用于存储所有的频道和模式。
-
pubsub_channels
:这是一个字典,键是频道名,值是一个链表,链表中存储了所有订阅了这个频道的客户端。当有新消息发布到这个频道时,服务器会遍历这个链表,将消息发送给所有的客户端。 -
pubsub_patterns
:这是一个链表,存储了所有的模式。每个模式都是一个redisPubSubPattern
结构,包含了模式本身和订阅这个模式的客户端。当有新消息发布时,服务器会遍历这个链表,查找所有匹配的模式,并将消息发送给订阅了这些模式的客户端。
通过这种方式,Redis 实现了发布订阅模式,使得消息的发布和订阅变得非常高效。
3、发布订阅的命令
3.1、SUBSCRIBE命令
SUBSCRIBE
用于订阅一个或多个频道的信息。
命令格式:SUBSCRIBE channel [channel ...]
channel
:需要订阅的频道名称,可以是一个或多个。
当客户端发送 SUBSCRIBE
命令订阅一个或多个频道时,服务器会接收到这个命令,并将这些频道添加到客户端的订阅列表中。然后,服务器会向客户端返回一个消息,确认已经订阅了这些频道。
一旦客户端订阅了一个频道,它就会持续监听这个频道,直到客户端发送 UNSUBSCRIBE
命令取消订阅,或者客户端断开连接。
当有新消息发布到这个频道时,服务器会将这个消息发送给所有订阅了这个频道的客户端。
例如,客户端可以发送如下命令订阅名为 mychannel
的频道:
SUBSCRIBE mychannel
服务器会返回如下消息:
1) "subscribe"
2) "mychannel"
3) (integer) 1
这表示客户端已经成功订阅了 mychannel
频道。
3.2、UNSUBSCRIBE命令
UNSUBSCRIBE
是 Redis 发布订阅模式中的一个命令,用于退订一个或多个频道的信息。
命令格式:UNSUBSCRIBE [channel [channel ...]]
channel
:需要退订的频道名称,可以是一个或多个。如果没有指定频道,则会退订所有频道。
当客户端发送 UNSUBSCRIBE
命令退订一个或多个频道时,服务器会接收到这个命令,并将这些频道从客户端的订阅列表中移除。然后,服务器会向客户端返回一个消息,确认已经退订了这些频道。
例如,客户端可以发送如下命令退订名为 mychannel
的频道:
UNSUBSCRIBE mychannel
服务器会返回如下消息:
1) "unsubscribe"
2) "mychannel"
3) (integer) 0
这表示客户端已经成功退订了 mychannel
频道。如果客户端再次收到 mychannel
频道的消息,那么这些消息将会被忽略。
3.3、PUBLISH命令
PUBLISH
是 Redis 发布订阅模式中的一个命令,用于将消息发送到指定的频道。
命令格式:PUBLISH channel message
channel
:消息需要发送到的频道名称。message
:需要发送的消息内容。
当客户端发送 PUBLISH
命令发布消息到某个频道时,服务器会接收到这个命令,并将消息发送给所有订阅了这个频道的客户端。
例如,客户端可以发送如下命令向名为 mychannel
的频道发布一条消息:
PUBLISH mychannel "hello"
服务器会返回一个整数,表示消息成功发送到的客户端数量。例如:
(integer) 1
这表示消息已经成功发送到了 1 个客户端。如果没有客户端订阅这个频道,那么这个命令将不会有任何效果。
相关文章:

Redis发布订阅
在现代的软件开发中,数据存储和管理是至关重要的一环。Redis,作为一个开源的、内存中的数据结构存储系统,以其出色的性能和灵活的数据结构,赢得了开发者们的广泛喜爱。它不仅可以用作数据库,还可以用作缓存和消息代理。…...

在Windows操作系统上安装PostgreSQL数据库
在Windows操作系统上安装PostgreSQL数据库 一、在Windows操作系统上安装PostgreSQL数据库 一、在Windows操作系统上安装PostgreSQL数据库 点击 PostgreSQL可跳转至PostGreSQL的官方下载地址。 (1) (2)选择安装的目录ÿ…...

【云原生】Kubeadmin部署Kubernetes集群
目录 编辑 一、环境准备 1.2调整内核参数 二、所有节点部署docker 三、所有节点安装kubeadm,kubelet和kubectl 3.1定义kubernetes源 3.2开机自启kubelet 四、部署K8S集群 4.1查看初始化需要的镜像 4.2在 master 节点上传 v1.20.11.zip 压缩包至 /opt 目录…...

Java中wait和notify详解
线程的调度是无序的,随机的,但是也是有一定的需求场景,希望能够有序执行,join算是一种控制顺序的方式(功能有限)——》一个线程执行完,才能执行另一个线程! 本文主要讲解的…...

算法竞赛个人注意事项
浅浅记录一下自己在算法竞赛中的注意事项。 数据类 注意看数大小,数学库中的函数尽量加上 * 1.0,转成double,防止整型溢出。,int型相乘如果可能溢出,乘 * 1LL。 数据范围大于1e6,注意用快读。 浮点数输…...
ClickHouse和Doris超大数据集存储
文章目录 一. ClickHouse1. 性能2. 可靠性3. 可扩展性4. 支持SQL和复杂查询5. 适用场景 二. Doris1. 性能2. 可靠性3. 易用性4. 适用场景 三. ClickHouse和Doris的比较1. 架构2. 性能3. 可靠性4. 易用性5. 适用场景 四. 总结 ClickHouse和Doris是两种流行的超大数据集存储方案。…...

02-Flask-对象初始化参数
对象初始化参数 前言对象初始化参数import_namestatic_url_pathstatic_foldertemplate_floder 前言 本篇来学习Flask中对象初始化参数 对象初始化参数 import_name Flask程序所在的包(模块),传__name__就可以 _name_ 是一个标识 Python 模块的名字的变量&#x…...

第5篇 vue的通信框架axios和ui框架-element-ui以及node.js
一 axios的使用 1.1 介绍以及作用 axios是独立于vue的一个项目,基于promise用于浏览器和node.js的http客户端。 在浏览器中可以帮助我们完成 ajax请求的发送在node.js中可以向远程接口发送请求 1.2 案例使用axios实现前后端数据交互 1.后端代码 2.前端代码 &…...

RabbitMQ 知识点解读
1、AMQP 协议 1.1、AMQP 生产者的流转过程 当客户端与Broker 建立连接的时候,会调用factory .newConnection 方法,这个方法会进一步封装成Protocol Header 0-9-1 的报文头发送给Broker ,以此通知Broker 本次交互采用的是AMQPO-9-1 协议&…...

SimVODIS++: Neural Semantic Visual Odometry in Dynamic Environments 论文阅读
论文信息 题目:SimVODIS: Neural Semantic Visual Odometry in Dynamic Environments 作者:Ue-Hwan Kim , Se-Ho Kim , and Jong-Hwan Kim , Fellow, IEEE 时间:2022 来源: IEEE ROBOTICS AND AUTOMATION LETTERS(RAL…...

7.Xaml Image控件
1.运行图片 2.运行源码 a.xaml源码 <!--Source="/th.gif" 图像源--><!--Stretch="Fill" 填充模式--><Image x:Name...

Solidity 小白教程:11. 构造函数和修饰器
Solidity 小白教程:11. 构造函数和修饰器 这一讲,我们将用合约权限控制(Ownable)的例子介绍solidity语言中构造函数(constructor)和独有的修饰器(modifier)。 构造函数 构造函数&…...

静态工厂模式,抽象工厂模式,建造者模式
静态工厂模式 ublic class FruitFactory {public static Fruit getFruit(String name) {Fruit fnull;switch (name){case "APPLE":{fnew Apple();}case "BANANA":{fnew Banana();}default :{System.out.println("Unknown Fruit");}}return f;} …...

【动手学深度学习笔记】--门控循环单元GRU
文章目录 门控循环单元GRU1.门控隐状态1.1重置门和更新门1.2候选隐状态1.3隐状态 2.从零开始实现2.1读取数据2.2初始化模型参数2.3定义模型2.4训练与预测 3.简洁实现 门控循环单元GRU 学习视频:门控循环单元(GRU)【动手学深度学习v2】 官方…...

浅析linux异步io框架 io_uring
前言 Linux内核5.1支持了新的异步IO框架iouring,由Block IO大神也即Fio作者Jens Axboe开发,意在提供一套公用的网络和磁盘异步IO,不过io_uring目前在磁盘方面要比网络方面更加成熟。 目录 背景简介 io_uring 系统API liburing 高级特性…...
访问者模式的一个使用案例——文档格式转换
访问者模式的一个使用案例——文档格式转换 假设我们在开发一个文档编辑器,它支持多种不同的文档元素(如段落、图片、表格等),现在我们需要添加一个功能——将文档导出为 HTML 或 Markdown 格式。 这就是一个典型的访问者模式的…...

【MySql】数据库的聚合查询
写在最前面的话 哈喽,宝子们,今天给大家带来的是MySql数据库的聚合查询。在前面CRUD章节我们学习了表达式查询,表达式查询是针对列和列之间进行运算的,那么如果想在行和行之间进行运算,那么就需要用到聚合查询。聚合查…...

Linux初探 - 概念上的理解和常见指令的使用
目录 Linux背景 Linux发展史 GNU 应用场景 发行版本 从概念上认识Linux 操作系统的概念 用户的概念 路径与目录 Linux下的文件 时间戳的概念 常规权限 特殊权限 Shell的概念 常用指令 ls tree stat clear pwd echo cd touch mkdir rmdir rm cp mv …...
苹果上架Guideline 4.3 - Design
最近上架苹果商店,审核提示 Guideline 4.3 - DesignWe noticed your app shares a similar binary, metadata, and/or concept as apps previously submitted by a terminated Apple Developer Program account.Submitting similar or repackaged apps is a form o…...

【数据分析入门】【淘宝电商API接入与电商数据分析】初识Web API(一)
今天开始我们将学习如何使用Web应用变成借口(API)自动请求网站到特定信息而不是整个网站,再对这些信息进行可视化。由于这样编写到程序始终使用最新到数据来生成可视化,因此即便数据瞬息万变,它呈现到信息也都是最新的。比如,我们…...

Mybatis逆向工程,动态创建实体类、条件扩展类、Mapper接口、Mapper.xml映射文件
今天呢,博主的学习进度也是步入了Java Mybatis 框架,目前正在逐步杨帆旗航。 那么接下来就给大家出一期有关 Mybatis 逆向工程的教学,希望能对大家有所帮助,也特别欢迎大家指点不足之处,小生很乐意接受正确的建议&…...

苍穹外卖--缓存菜品
1.问题说明 用户端小程序展示的菜品数据都是通过查询数据库获得,如果用户端访问量比较大,数据库访问压力随之增大 2.实现思路 通过Redis来缓存菜品数据,减少数据库查询操作。 缓存逻辑分析: ①每个分类下的菜品保持一份缓存数据…...

Module Federation 和 Native Federation 的比较
前言 Module Federation 是 Webpack 5 引入的微前端架构方案,允许不同独立构建的应用在运行时动态共享模块。 Native Federation 是 Angular 官方基于 Module Federation 理念实现的专为 Angular 优化的微前端方案。 概念解析 Module Federation (模块联邦) Modul…...

自然语言处理——Transformer
自然语言处理——Transformer 自注意力机制多头注意力机制Transformer 虽然循环神经网络可以对具有序列特性的数据非常有效,它能挖掘数据中的时序信息以及语义信息,但是它有一个很大的缺陷——很难并行化。 我们可以考虑用CNN来替代RNN,但是…...

ios苹果系统,js 滑动屏幕、锚定无效
现象:window.addEventListener监听touch无效,划不动屏幕,但是代码逻辑都有执行到。 scrollIntoView也无效。 原因:这是因为 iOS 的触摸事件处理机制和 touch-action: none 的设置有关。ios有太多得交互动作,从而会影响…...

均衡后的SNRSINR
本文主要摘自参考文献中的前两篇,相关文献中经常会出现MIMO检测后的SINR不过一直没有找到相关数学推到过程,其中文献[1]中给出了相关原理在此仅做记录。 1. 系统模型 复信道模型 n t n_t nt 根发送天线, n r n_r nr 根接收天线的 MIMO 系…...

C++_哈希表
本篇文章是对C学习的哈希表部分的学习分享 相信一定会对你有所帮助~ 那咱们废话不多说,直接开始吧! 一、基础概念 1. 哈希核心思想: 哈希函数的作用:通过此函数建立一个Key与存储位置之间的映射关系。理想目标:实现…...

JDK 17 序列化是怎么回事
如何序列化?其实很简单,就是根据每个类型,用工厂类调用。逐个完成。 没什么漂亮的代码,只有有效、稳定的代码。 代码中调用toJson toJson 代码 mapper.writeValueAsString ObjectMapper DefaultSerializerProvider 一堆实…...
用递归算法解锁「子集」问题 —— LeetCode 78题解析
文章目录 一、题目介绍二、递归思路详解:从决策树开始理解三、解法一:二叉决策树 DFS四、解法二:组合式回溯写法(推荐)五、解法对比 递归算法是编程中一种非常强大且常见的思想,它能够优雅地解决很多复杂的…...

VSCode 使用CMake 构建 Qt 5 窗口程序
首先,目录结构如下图: 运行效果: cmake -B build cmake --build build 运行: windeployqt.exe F:\testQt5\build\Debug\app.exe main.cpp #include "mainwindow.h"#include <QAppli...