RabbitMQ 确认模式(Acknowledgements Mode)详解

        RabbitMQ 是一款流行的开源消息代理软件,它实现了高级消息队列协议(AMQP)。在消息传递过程中,确保消息被正确处理是至关重要的。RabbitMQ 提供了多种机制来确保消息的可靠性,其中确认模式(Acknowledgements Mode)是一个关键特性。

什么是确认模式?

        确认模式(Acknowledgements Mode)允许消费者在成功处理消息后显式地向 RabbitMQ 服务器发送确认信号(ack)。只有在收到确认信号后,RabbitMQ 服务器才会从队列中删除该消息。如果消费者未能发送确认信号(例如,由于消费者崩溃或网络故障),RabbitMQ 会认为消息尚未被处理,并在适当的时候重新发送消息。

RabbitMQ 提供了三种主要的确认模式:

  1. 手动确认(Manual Acknowledgement):消费者需要显式地发送确认信号。
  2. 自动确认(Automatic Acknowledgement):消息一旦被消费者接收,立即自动确认。
  3. 批量确认(Batch Acknowledgement):消费者可以一次确认多条消息。
为什么使用确认模式?
  • 确保消息不丢失:即使消费者崩溃,消息也会重新发送。
  • 提高可靠性:通过控制确认时机,可以更好地管理消息处理流程。
  • 灵活性:可以根据不同的业务需求选择不同的确认模式。
Java 代码示例

        下面是一个使用 Java 和 Spring AMQP 实现 RabbitMQ 确认模式的示例。

Maven 依赖

        首先,在你的 pom.xml 文件中添加 Spring AMQP 依赖:

<dependencies>  <dependency>  <groupId>org.springframework.boot</groupId>  <artifactId>spring-boot-starter-amqp</artifactId>  </dependency>  <!-- 其他依赖 -->  
</dependencies>
配置 RabbitMQ

        在 application.properties 文件中配置 RabbitMQ 连接信息:

spring.rabbitmq.host=localhost  
spring.rabbitmq.port=5672  
spring.rabbitmq.username=guest  
spring.rabbitmq.password=guest
配置类

        配置一个队列、交换机和绑定:

import org.springframework.amqp.core.*;  
import org.springframework.context.annotation.Bean;  
import org.springframework.context.annotation.Configuration;  @Configuration  
public class RabbitMQConfig {  public static final String QUEUE_NAME = "exampleQueue";  public static final String EXCHANGE_NAME = "exampleExchange";  public static final String ROUTING_KEY = "exampleRoutingKey";  @Bean  public Queue queue() {  return new Queue(QUEUE_NAME, true); // 持久化队列  }  @Bean  public DirectExchange exchange() {  return new DirectExchange(EXCHANGE_NAME);  }  @Bean  public Binding binding(Queue queue, DirectExchange exchange) {  return BindingBuilder.bind(queue).to(exchange).with(ROUTING_KEY);  }  
}
消息监听器

        使用手动确认模式来监听队列:

import org.springframework.amqp.rabbit.annotation.RabbitListener;  
import org.springframework.amqp.rabbit.core.ChannelAwareMessageListener;  
import org.springframework.amqp.rabbit.core.RabbitTemplate;  
import org.springframework.amqp.rabbit.listener.api.ChannelAwareMessageListener.ReturnCallback;  
import org.springframework.amqp.rabbit.listener.api.RabbitListenerErrorHandler;  
import org.springframework.amqp.rabbit.listener.config.SimpleRabbitListenerContainerFactory;  
import org.springframework.amqp.rabbit.listener.config.SimpleRabbitListenerEndpoint;  
import org.springframework.amqp.rabbit.connection.ConnectionFactory;  
import org.springframework.amqp.rabbit.listener.AbstractMessageListenerContainer;  
import org.springframework.amqp.support.converter.MessageConverter;  
import org.springframework.amqp.support.converter.SimpleMessageConverter;  
import org.springframework.beans.factory.annotation.Autowired;  
import org.springframework.context.annotation.Bean;  
import org.springframework.context.annotation.Configuration;  import com.rabbitmq.client.Channel;  @Configuration  
public class ListenerConfig {  @Autowired  private ConnectionFactory connectionFactory;  @Bean  public SimpleRabbitListenerContainerFactory rabbitListenerContainerFactory() {  SimpleRabbitListenerContainerFactory factory = new SimpleRabbitListenerContainerFactory();  factory.setConnectionFactory(connectionFactory);  factory.setMessageConverter(jsonMessageConverter());  factory.setAcknowledgeMode(AcknowledgeMode.MANUAL); // 设置为手动确认模式  return factory;  }  @Bean  public MessageConverter jsonMessageConverter() {  return new SimpleMessageConverter();  }  @RabbitListener(queues = RabbitMQConfig.QUEUE_NAME, containerFactory = "rabbitListenerContainerFactory")  public void listen(String message, Channel channel, org.springframework.amqp.core.Message rabbitMessage) throws Exception {  try {  // 处理消息  System.out.println("Received <" + message + ">");  // 发送确认信号  channel.basicAck(rabbitMessage.getMessageProperties().getDeliveryTag(), false);  } catch (Exception e) {  // 发送拒绝信号,并设置为重新入队(requeue)  channel.basicNack(rabbitMessage.getMessageProperties().getDeliveryTag(), false, true);  throw e;  }  }  
}
发送消息

        编写一个简单的控制器来发送消息到 RabbitMQ:

import org.springframework.amqp.rabbit.core.RabbitTemplate;  
import org.springframework.beans.factory.annotation.Autowired;  
import org.springframework.web.bind.annotation.GetMapping;  
import org.springframework.web.bind.annotation.RequestParam;  
import org.springframework.web.bind.annotation.RestController;  @RestController  
public class MessageController {  @Autowired  private RabbitTemplate rabbitTemplate;  @GetMapping("/send")  public String sendMessage(@RequestParam String message) {  rabbitTemplate.convertAndSend(RabbitMQConfig.EXCHANGE_NAME, RabbitMQConfig.ROUTING_KEY, message);  return "Message sent: " + message;  }  
}
运行示例
  1. 启动 Spring Boot 应用程序。
  2. 使用浏览器或工具(如 Postman)访问 http://localhost:8080/send?message=HelloRabbitMQ 来发送消息。
  3. 观察控制台输出,确认消息已被接收并处理。

通过这种方式,你可以确保消息在成功处理后才会从队列中删除,从而提高了消息传递的可靠性。


新时代农民工

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/458344.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【Web.路由】——路由原理

这篇文章&#xff0c;我们来讲一讲什么是路由。 路由是 将用户请求地址映射为一个请求委托的过程&#xff0c;负责匹配传入的Http请求&#xff0c;然后将这些请求发送到应用的可执行终结点。 这里需要注意一个内容&#xff0c;发送到应用的可执行终结点。 路由的分类&#x…

大模型落地,要追求极致的务实主义

图源&#xff1a;AI生成 ▎****更快用上最新的大模型&#xff0c;是不是就赢定了&#xff1f; “能够像人类一样操作电脑。”这一堪称革命性的新技能来自10月23日Anthropic最新推出的升级版Claude 3.5 Sonnet模型&#xff0c;据介绍该模型不仅各项性能指标上取得显著提升&…

MobileNetV2实现实时口罩检测tensorflow

项目源码获取方式见文章末尾&#xff01; 回复暗号&#xff1a;13&#xff0c;免费获取600多个深度学习项目资料&#xff0c;快来加入社群一起学习吧。 **《------往期经典推荐------》**项目名称 1.【Informer模型复现项目实战】 2.【卫星图像道路检测DeepLabV3Plus模型】 3.【…

机器学习 - 概述、数据集、Scikit-learn

目录 一、人工智能概述1、概念2、应用领域 二、机器学习1、概念2、数据集的构成3、算法分类a、监督学习b、无监督学习 4、开发流程5、可用数据集总结 三、Scikit-learn1、介绍2、安装3、数据集API介绍4、数据集返回值介绍5、数据集的划分6、数据集划分的API介绍7、案例a、获取鸢…

sass软件数据架构思路——未来之窗行业应用跨平台架构

一、SAAS多商户数据库 1.1 SaaS 多商户数据库的全部商户数据放在一个服务器上 1. 成本效益&#xff1a;集中管理和维护一个服务器通常比维护多个独立服务器更经济&#xff0c;降低硬件、运维和管理成本。 2. 数据集中管理&#xff1a;便于进行统一的数据备份、恢复和数据治理…

高清 MV 无字幕视频素材

在当下的短视频和自媒体时代&#xff0c;高清无字幕的视频素材无疑是创作者们的“得力助手”。不管是用于剪辑情感励志视频、制作搞笑段子&#xff0c;还是创作风景航拍视频&#xff0c;优质的素材库都能让你的创作如虎添翼。今天&#xff0c;我就为大家介绍几个海外的高质量素…

光学计算领域的突破:更快、更高效的光子存储单元

国际电气工程师团队首次开发出一种新的光子内存计算方法&#xff0c;这将在不久的将来使光学计算成为现实。 该团队包括来自匹兹堡大学斯旺森工程学院、加州大学圣巴巴拉分校、卡利亚里大学和东京工业大学&#xff08;现东京科学大学&#xff09;的研究人员。他们的研究成果发…

【论文阅读】Reliable, Adaptable, and Attributable Language Models with Retrieval

文章目录 OverviewCurrent Retrieval-Augmented LMsArchitectureTraining Limitations & Future Work Overview Parametic language models的缺点&#xff1a; 事实性错误的普遍存在验证的难度&#xff08;可溯源性差&#xff09;难以在有顾虑的情况下排除某些序列适应调整…

web pdf 图片拖动图片合成

web pdf 图片拖动图片合成 先看效果 前端 合成后 1.原理 以前写过相关的帖子&#xff0c;使用的是 canva 但是这次换了一个思路使用的是图片 1.先把pdf转成图片 2.把pdf图片和目标图片传到浏览器 3.原理就和图片合成一样了。见上一篇帖子 4.后端也一样只不过这次是将位置和pd…

LeetCode 107.二叉树的层次遍历 II

题目描述 给你二叉树的根节点 root &#xff0c;返回其节点值 自底向上的层序遍历 。 &#xff08;即按从叶子节点所在层到根节点所在的层&#xff0c;逐层从左向右遍历&#xff09; 示例 1&#xff1a; 输入&#xff1a;root [3,9,20,null,null,15,7] 输出&#xff1a;[[1…

【力扣专题栏】两数相加,如何实现存储在链表中的整数相加?

题解目录 1、题目描述解释2、算法原理解析3、代码编写&#xff08;原始版本&#xff09;4、代码编写&#xff08;优化版本&#xff09; 1、题目描述解释 2、算法原理解析 3、代码编写&#xff08;原始版本&#xff09; /*** Definition for singly-linked list.* struct ListN…

快速上手机器学习-朴素贝叶斯

朴素贝叶斯 引言&#xff1a;本文通过介绍先验概率&#xff0c;后验概率&#xff0c;条件概率计算和贝叶斯定理等概率论内容引入朴素贝叶斯分类算法的基本思路&#xff0c;朴素贝叶斯的最终分类思想是将输入分类给概率最大的类&#xff0c;这也是概率模型算法的共有思想。本文专…

JVM(HotSpot):GC之G1垃圾回收器

文章目录 一、简介二、工作原理三、Young Collection 跨代引用四、大对象问题 一、简介 1、适用场景 同时注重吞吐量&#xff08;Throughput&#xff09;和低延迟&#xff08;Low latency&#xff09;&#xff0c;默认的暂停目标是 200 ms超大堆内存&#xff0c;会将堆划分为…

第12次CCF CSP认证真题解

1、最小差值 题目链接&#xff1a;https://sim.csp.thusaac.com/contest/12/problem/0 100分代码&#xff1a; #include <iostream> #include <algorithm> using namespace std; int main(int argc, char *argv[]) {int n;cin >> n;int a[1010],b[1010];f…

【读书笔记·VLSI电路设计方法解密】问题28:什么是芯片可靠性

一块VLSI芯片不仅需要在前几天、几周或几个月内正常工作,还必须在整个使用寿命内可靠运行。任何用于商业目的的芯片的寿命通常定义为100,000小时或约11.4年。然而,在IC的设计、制造、组装和测试过程中,许多因素可能导致其早期失效。这一芯片开发方面被称为芯片可靠性。测试相…

LinkedList和链表之刷题课(下)

1. 给定x根据x把链表分割,大的结点放在x后面,小的结点放在x前面 题目解析: 注意此时的pHead就是head(头节点的意思) 基本上就是给定一个链表,我们根据x的值来把这个链表分成俩部分,大的那部分放在x后面,小的那部分放在x前面,并且我们不能改变链表本来的顺序,比如下面的链表,我…

【数据结构与算法】《Java 算法宝典:探秘从排序到回溯的奇妙世界》

目录 标题&#xff1a;《Java 算法宝典&#xff1a;探秘从排序到回溯的奇妙世界》一、排序算法1、冒泡排序2、选择排序3、插入排序4、快速排序5、归并排序 二、查找算法1、线性查找2、二分查找 三、递归算法四、动态规划五、图算法1. 深度优先搜索&#xff08;DFS&#xff09;2…

Ubuntu22.04环境搭建MQTT服务器

官网&#xff1a; https://mosquitto.org 1.引入库 sudo apt-add-repository ppa:mosquitto-dev/mosquitto-ppa2.升级安装工具 sudo apt-get update 3.安装 sudo apt-get install mosquitto 4.安装客户端 sudo apt-get install mosquitto-clients5.添加修改配置文件 进…

MySql数据库中数据类型

本篇将介绍在 MySql 中的所有数据类型&#xff0c;其中主要分为四类&#xff1a;数值类型、文本和二进制类型、时间日期、String 类型。如下&#xff08;图片来源&#xff1a;MySQL数据库&#xff09;&#xff1a; 目录如下&#xff1a; 目录 数值类型 1. 整数类型 2. …

Python | Leetcode Python题解之第516题最长回文子序列

题目&#xff1a; 题解&#xff1a; class Solution:def longestPalindromeSubseq(self, s: str) -> int:n len(s)dp [[0] * n for _ in range(n)]for i in range(n - 1, -1, -1):dp[i][i] 1for j in range(i 1, n):if s[i] s[j]:dp[i][j] dp[i 1][j - 1] 2else:dp…