利用Redis的队列模式实现消息的发送和订阅,适合分布式场景,Java实现代码
在Redis中,通常使用发布/订阅模式(Pub/Sub)来进行消息的实时通信。然而,标准的Redis发布/订阅模式并不直接支持确保一条消息只被一台机器消费。在这种模式下,所有订阅了特定频道的客户端都会收到发布的消息。
但是,你可以通过一些策略或模式来模拟这种“只在一台机器上消费”的行为。以下是一些可能的方法:
1. 使用Redis的分布式锁
发布消息:当消息发布时,使用一个Redis的分布式锁(如RedLock)来确保只有一个消费者能够处理该消息。
处理消息:消费者尝试获取锁。如果成功,则处理消息并释放锁;如果失败,则放弃处理该消息。
2. 队列模式
发布消息:不是直接将消息发布到频道,而是将消息推送到一个Redis列表(List)或有序集合(Sorted Set)中。
消费消息:消费者使用BLPOP、BRPOP或其他阻塞操作从列表中拉取消息。由于这些操作是阻塞的,因此它们会等待直到有消息可用。同时,由于只有一个消费者能够成功地从列表中拉取消息,因此可以确保消息只被一台机器消费。
3. 分布式任务队列
使用更高级的分布式任务队列系统(如Celery、RabbitMQ、Kafka等),这些系统通常提供了更复杂的路由和消息确认机制,可以确保消息只被一台机器消费。
4. 自定义发布/订阅逻辑
在应用层实现自定义的发布/订阅逻辑。例如,你可以使用一个Redis哈希(Hash)来跟踪哪些消息已经被哪台机器消费。当发布消息时,检查哈希以确定是否有机器已经消费了该消息。如果没有,则选择一个消费者并将消息发送到其队列中。
注意事项:
在实现这些策略时,请确保考虑并发性和性能问题。例如,分布式锁和Redis列表操作在高并发环境下可能会成为瓶颈。
确保你的Redis实例是可靠和可扩展的。在分布式系统中,Redis实例可能会成为单点故障或性能瓶颈。考虑使用Redis集群或其他高可用性解决方案来避免这些问题。
根据你的具体需求和场景选择合适的策略。例如,如果你的系统需要处理大量实时消息并且需要确保消息的顺序性,那么使用Redis列表和阻塞操作可能是个好主意。但是,如果你的系统需要更复杂的路由和消息确认机制,那么使用分布式任务队列系统可能更合适。
下面介绍用队列模式,简单实现队列如下:
要使用Redis的队列模式实现消息的消费,我们可以使用Jedis库(Java的Redis客户端)来与Redis进行交互。以下是一个简单的Java实现,展示了如何使用Redis的List数据结构(通常使用LPUSH和BRPOP命令)来实现消息的发布和消费。
首先,确保你已经添加了Jedis的依赖到你的项目中。如果你使用Maven,可以在pom.xml中添加以下依赖:
xml
复制
<dependency>
<groupId>redis.clients</groupId>
<artifactId>jedis</artifactId>
<version>3.7.0</version> <!-- 请检查并使用最新版本 -->
</dependency>
然后,你可以编写两个简单的Java类,一个用于发布消息(Producer),另一个用于消费消息(Consumer)。
Producer(消息发布者)
java
复制
import redis.clients.jedis.Jedis;
public class RedisMessageProducer {
private final Jedis jedis;
private final String queueName;
public RedisMessageProducer(Jedis jedis, String queueName) {
this.jedis = jedis;
this.queueName = queueName;
}
public void produce(String message) {
jedis.lpush(queueName, message);
System.out.println("Produced: " + message);
}
public static void main(String[] args) {
Jedis jedis = new Jedis("localhost"); // 假设Redis服务器运行在本地
RedisMessageProducer producer = new RedisMessageProducer(jedis, "my-queue");
// 发布消息
producer.produce("Hello, Redis Queue!");
producer.produce("Another message for the queue.");
// 关闭连接(在实际应用中,你可能希望使用连接池来管理连接)
jedis.close();
}
}
Consumer(消息消费者)
java
复制
import redis.clients.jedis.Jedis;
public class RedisMessageConsumer implements Runnable {
private final Jedis jedis;
private final String queueName;
public RedisMessageConsumer(Jedis jedis, String queueName) {
this.jedis = jedis;
this.queueName = queueName;
}
@Override
public void run() {
while (true) { // 无限循环,直到应用程序被终止
String message = jedis.brpop(0, queueName).get(1); // 阻塞直到有消息可用
if (message != null) {
System.out.println("Consumed: " + message);
// 在这里处理消息...
}
}
}
public static void main(String[] args) {
Jedis jedis = new Jedis("localhost"); // 假设Redis服务器运行在本地
RedisMessageConsumer consumer = new RedisMessageConsumer(jedis, "my-queue");
// 在新的线程中运行消费者
new Thread(consumer).start();
// 注意:这里的main方法不会立即结束,因为消费者在一个无限循环中运行。
// 在实际应用中,你可能希望以不同的方式管理消费者的生命周期。
}
}
注意:
在这个例子中,RedisMessageConsumer的main方法启动了一个新线程来运行消费者。在实际应用中,你可能希望以更复杂的方式管理这些线程,例如使用线程池或Spring的@Async注解。
jedis.brpop(0, queueName)中的0表示阻塞的时间(以秒为单位)。传递0意味着它将无限期地阻塞,直到有消息可用。
请确保你的Redis服务器正在运行,并且Java应用程序可以访问它。如果Redis服务器不在本地运行,你需要将Jedis的构造函数中的"localhost"替换为Redis服务器的实际地址。
在实际应用中,你可能还需要处理异常、优雅地关闭连接以及确保在应用程序终止时正确地清理资源。
相关文章:
利用Redis的队列模式实现消息的发送和订阅,适合分布式场景,Java实现代码
在Redis中,通常使用发布/订阅模式(Pub/Sub)来进行消息的实时通信。然而,标准的Redis发布/订阅模式并不直接支持确保一条消息只被一台机器消费。在这种模式下,所有订阅了特定频道的客户端都会收到发布的消息。 但是&…...
软件下载安装【汇总】
软件下载安装【汇总】 前言版权推荐软件安装【汇总】最后 前言 2024-5-12 21:38:34 以下内容源自《【汇总】》 仅供学习交流使用 版权 禁止其他平台发布时删除以下此话 本文首次发布于CSDN平台 作者是CSDN日星月云 博客主页是https://jsss-1.blog.csdn.net 禁止其他平台发布…...
重定向文件访问(Redirect file access)
重定向文件访问 重定向文件访问是指通过修改文件系统的路径,使对某个文件或目录的访问请求被转到另一个文件或目录。这在系统管理、测试和开发中非常有用,因为它允许您在不修改应用程序或服务配置的情况下,改变文件的实际存储位置。 proot …...
隐私计算(1)数据可信流通
目录 1. 数据可信流通体系 2. 信任的基石 3.数据流通中的不可信风险 可信链条的级联失效,以至于崩塌 4.数据内循环与外循环:传统数据安全的信任基础 4.1内循环 4.2外循环 5. 技术信任 6. 密态计算 7.技术信任 7.1可信数字身份 7.2 使用权跨域…...
果汁机锂电池充电,5V升压12.7V 升压恒压芯片SL1571B
在现代化的日常生活中,果汁机已经逐渐成为了许多家庭厨房的必备电器。随着科技的不断进步,果汁机的性能也在不断提升,其中锂电池的应用更是为果汁机带来了前所未有的便利。而5V升压12.7V升压恒压芯片SL1571B,作为果汁机锂电池充电…...
多个线程多个锁:如何确保线程安全和避免竞争条件
目录 前言 一、确定需要多个锁的场景 1.独立资源保护 2.部分依赖资源 二、避免死锁 三、锁粒度与并发性能 1. 粗粒度锁定 2.细粒度锁定 四、设计策略:减少资源依赖 1.资源分离 2.无锁设计 3.锁合并 五、Demo讲解 总结: 前言 当多个线程需要…...
Linux-笔记 设备树插件
目录 前言: 设备树插件的书写规范: 设备树插件的编译: 内核配置: 应用背景: 举例: 前言: 设备树插件(Device Tree Blob Overlay,简称 DTBO)是Linux内核和嵌入式系统…...
【排序算法】总结篇
✨✨这些 排序算法都是指的 需要进行比较的排序算法 ✨✨下面都是略微讲解一下思路,如果需要详细了解哪一个排序,点击👉链接即可 ✨✨对于时间、空间复杂度、稳定性,希望你🧑🎓能够理解记忆🧑…...
鸿蒙开发文件管理:【@ohos.fileio (文件管理)】
文件管理 该模块提供文件存储管理能力,包括文件基本管理、文件目录管理、文件信息统计、文件流式读写等常用功能。 说明: 本模块首批接口从API version 6开始支持。后续版本的新增接口,采用上角标单独标记接口的起始版本。 导入模块 impor…...
硬件工程师学习规划
背景介绍 当前电子行业中,互联网因为中国人口基数大,得到很快的发展,一越成为世界第一梯队,互联网软件薪资要高于传统制造业硬件的薪资,从各大招聘软件上就能看到,那么为什么软件发展要好于硬件࿱…...
esp32 8行代码实现蓝牙音响
目录 硬件准备: 具体代码: 接线: 备注: 八行代码实现简易版蓝牙音响,亲测有效: esp32 DIY蓝牙音响_哔哩哔哩_bilibili 硬件准备: ESP32-wroom、MAX98357音频放大器模块、4欧3瓦小喇叭、杜…...
注册用户如何防止缓存穿透?
注册用户如何防止缓存穿透? 先说明用户注册为什么会发送缓存穿透:用户注册时,需要验证用户名是否已存在,先查缓存,没有再查数据库,还没有才验证通过。高并发的情况下就可能有大量用户同时注册,…...
Presto基础知识
Presto缓存 引入Presto缓存之前 BackgroundHiveSplitLoader 使用底层的文件系统直接进行数据的读写; 引入Presto缓存机制之后,底层的文件系统被被CachingFileSystem 代理一层 CachingFileSystem 有两个子类,根据你选用的底层缓存引擎的不同…...
Ajax + Easy Excel 通过Blob实现导出excel
前端代码 <!DOCTYPE html> <html><head><meta charset"utf-8"><title></title><script src"./js/jquery-3.6.0.min.js"></script></head><body><div><button onclick"exportF…...
Qt+qss动态属性改变控件状态切换的样式
先说点基础的吧,qt的样式实现,常见的主要有三种方式,分别为: 1.ui界面中右键样式表直接添加 2.代码中对控件设置样式setStyleSheet 3.外部预设好qss文件,代码中加载后设置样式 实际工作开发中,我推荐使用优…...
纷享销客安全体系:安全运维运营
安全运维运营(Security Operations,SecOps)是指在信息安全管理中负责监控、检测、响应和恢复安全事件的一系列运营活动。它旨在保护组织的信息系统和数据免受安全威胁和攻击的损害。 通过有效的安全运维运营,组织可以及时发现和应对安全威胁,减少安全事…...
富瀚微FH8322 ISP图像调试—BLC校正
1、简单介绍 目录 1、简单介绍 2、调试方法 3、输出结果 富瀚微平台调试有一段时间了,一直没有总结,我们调试ISP的时候,首先一步时确定好sensor的黑电平值,黑电平如果不准,则会影响到后面的颜色及对比度相关模块。…...
什么是大型语言模型 ?
引言 在本文[1]中,我们将从高层次概述大型语言模型 (LLM) 的具体含义。 背景 2023年11月,我偶然间听闻了OpenAI的开发者大会,这个大会展示了人工智能领域的革命性进展,让我深深着迷。怀着对这一领域的浓厚兴趣,我加入了…...
RocketMq详解:二、SpringBoot集成RocketMq
在上一章中我们对Rocket的基础知识、特性以及四大核心组件进行了详细的介绍,本章带着大家一起去在项目中具体的进行应用,并设计将其作为一个工具包只提供消息的分发服务和业务模块进行解耦 在进行本章的学习之前,需要确保你的可以正常启动和…...
【源码】二开版微盘交易系统/贵金属交易平台/微交易系统
二开版微盘交易系统/贵金属交易平台/微交易系统 一套二开前端UI得贵金属微交易系统,前端产品后台可任意更换 此系统框架不是以往的至尊的框架,系统完美运行,K线采用nodejs方式运行 K线结算都正常,附带教程 资源来源:https://www.…...
大数据学习栈记——Neo4j的安装与使用
本文介绍图数据库Neofj的安装与使用,操作系统:Ubuntu24.04,Neofj版本:2025.04.0。 Apt安装 Neofj可以进行官网安装:Neo4j Deployment Center - Graph Database & Analytics 我这里安装是添加软件源的方法 最新版…...
ubuntu搭建nfs服务centos挂载访问
在Ubuntu上设置NFS服务器 在Ubuntu上,你可以使用apt包管理器来安装NFS服务器。打开终端并运行: sudo apt update sudo apt install nfs-kernel-server创建共享目录 创建一个目录用于共享,例如/shared: sudo mkdir /shared sud…...
基于距离变化能量开销动态调整的WSN低功耗拓扑控制开销算法matlab仿真
目录 1.程序功能描述 2.测试软件版本以及运行结果展示 3.核心程序 4.算法仿真参数 5.算法理论概述 6.参考文献 7.完整程序 1.程序功能描述 通过动态调整节点通信的能量开销,平衡网络负载,延长WSN生命周期。具体通过建立基于距离的能量消耗模型&am…...
centos 7 部署awstats 网站访问检测
一、基础环境准备(两种安装方式都要做) bash # 安装必要依赖 yum install -y httpd perl mod_perl perl-Time-HiRes perl-DateTime systemctl enable httpd # 设置 Apache 开机自启 systemctl start httpd # 启动 Apache二、安装 AWStats࿰…...
学习STC51单片机32(芯片为STC89C52RCRC)OLED显示屏2
每日一言 今天的每一份坚持,都是在为未来积攒底气。 案例:OLED显示一个A 这边观察到一个点,怎么雪花了就是都是乱七八糟的占满了屏幕。。 解释 : 如果代码里信号切换太快(比如 SDA 刚变,SCL 立刻变&#…...
Docker 本地安装 mysql 数据库
Docker: Accelerated Container Application Development 下载对应操作系统版本的 docker ;并安装。 基础操作不再赘述。 打开 macOS 终端,开始 docker 安装mysql之旅 第一步 docker search mysql 》〉docker search mysql NAME DE…...
CVE-2020-17519源码分析与漏洞复现(Flink 任意文件读取)
漏洞概览 漏洞名称:Apache Flink REST API 任意文件读取漏洞CVE编号:CVE-2020-17519CVSS评分:7.5影响版本:Apache Flink 1.11.0、1.11.1、1.11.2修复版本:≥ 1.11.3 或 ≥ 1.12.0漏洞类型:路径遍历&#x…...
[ACTF2020 新生赛]Include 1(php://filter伪协议)
题目 做法 启动靶机,点进去 点进去 查看URL,有 ?fileflag.php说明存在文件包含,原理是php://filter 协议 当它与包含函数结合时,php://filter流会被当作php文件执行。 用php://filter加编码,能让PHP把文件内容…...
Leetcode33( 搜索旋转排序数组)
题目表述 整数数组 nums 按升序排列,数组中的值 互不相同 。 在传递给函数之前,nums 在预先未知的某个下标 k(0 < k < nums.length)上进行了 旋转,使数组变为 [nums[k], nums[k1], …, nums[n-1], nums[0], nu…...
微服务通信安全:深入解析mTLS的原理与实践
🔥「炎码工坊」技术弹药已装填! 点击关注 → 解锁工业级干货【工具实测|项目避坑|源码燃烧指南】 一、引言:微服务时代的通信安全挑战 随着云原生和微服务架构的普及,服务间的通信安全成为系统设计的核心议题。传统的单体架构中&…...
