RabbitMQ高级特性详解

前言

RabbitMQ是一款广泛使用的开源消息队列软件,它基于AMQP(Advanced Message Queuing Protocol)标准实现。本文将带你深入了解RabbitMQ的一些高级特性,包括消息确认、死信队列、延迟队列、事务处理以及消息分发策略等,并通过示例代码展示如何在实际项目中应用这些特性。


目录

1. 消息确认机制

1.1 什么是消息确认?

自动确认 vs 手动确认

1.2 使用场景

2. 死信队列

2.1 死信的概念与来源

2.2 应用场景

3. 延迟队列

3.1 延迟队列简介

3.2 实现方法

示例代码

3.3 场景应用

4. 事务处理

4.1 事务的基本概念

4.2 配置与使用

4.3 生产者示例

5. 消息分发

5.1 分发机制

5.2 限流与负载均衡

5.3 示例代码

总结


1. 消息确认机制

1.1 什么是消息确认?

消息确认是确保消息从队列可靠地到达消费者的关键机制。生产者发送消息后,消息可能会被成功处理,也可能因为异常而未被正确处理。为了保证消息的可靠性,RabbitMQ引入了消息确认机制。

自动确认 vs 手动确认
  • 自动确认:当autoAck设置为true时,RabbitMQ会自动认为消息一旦发送出去就被消费端接收到了,并立刻从队列中删除这条消息。这种方式适合对消息可靠性要求不高的场景。
  • 手动确认:当autoAck设置为false时,RabbitMQ会在接收到消费者的显式确认信号后才会移除消息。这种模式适用于需要高可靠性的场景。
// 自动确认示例
channel.basicConsume("queueName", true, new DefaultConsumer(channel) {@Overridepublic void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {System.out.println("接收到消息: " + new String(body));}
});// 手动确认示例
channel.basicConsume("queueName", false, new DefaultConsumer(channel) {@Overridepublic void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {try {// 处理消息System.out.println("接收到消息: " + new String(body));// 确认消息channel.basicAck(envelope.getDeliveryTag(), false);} catch (Exception e) {// 异常处理,拒绝签收channel.basicNack(envelope.getDeliveryTag(), false, true); // requeue设为true表示重新入队}}
});

1.2 使用场景

使用自动确认可以简化开发流程,但可能会导致消息丢失。手动确认虽然增加了额外的确认步骤,但是能显著提高消息传递的可靠性,特别适用于金融交易、订单处理等关键业务场景。


2. 死信队列

2.1 死信的概念与来源

死信是指那些无法被正常消费的消息。它们可能由于以下几种情况产生:

  • 消息过期:消息在队列中的存活时间超过了设定的时间阈值(TTL)。
  • 消息被拒绝:消费者在处理过程中遇到错误并拒绝了该消息。
  • 队列满载:当队列达到最大长度限制时,新来的消息会被视为死信。

2.2 应用场景

  • 消息重试:将未能处理的消息重新发送到原始队列或另一个队列进行尝试。
  • 消息丢弃:直接丢弃那些不可处理的消息以避免占用资源。
  • 日志收集:将死信作为日志记录下来,以便后续分析问题所在。
@Bean
public Queue normalQueue() {Map<String, Object> arguments = new HashMap<>();arguments.put("x-dead-letter-exchange", "dlxExchange");arguments.put("x-dead-letter-routing-key", "dlxRoutingKey");return QueueBuilder.durable("normalQueue").withArguments(arguments).build();
}@RabbitListener(queues = "dlxQueue")
public void listenerDLXQueue(Message message, Channel channel) throws Exception {long deliveryTag = message.getMessageProperties().getDeliveryTag();System.out.printf("死信队列接收到消息: %s, deliveryTag: %d%n", new String(message.getBody()), deliveryTag);
}

3. 延迟队列

3.1 延迟队列简介

延迟队列是一种特殊的队列,其中的消息不是立即被消费,而是等待指定的时间后才可被获取。RabbitMQ本身并不支持延迟队列功能,但可以通过结合TTL(Time to Live)和死信队列来模拟实现。

3.2 实现方法

  • TTL+死信队列:给消息设置生存时间,超时后转至死信队列。
  • 官方插件:使用RabbitMQ提供的延迟消息插件。
示例代码
@Bean
public Queue delayedQueue() {Map<String, Object> args = new HashMap<>();args.put("x-dead-letter-exchange", "dlxExchange");args.put("x-message-ttl", 5000); // 设置消息TTL为5秒return QueueBuilder.durable("delayedQueue").withArguments(args).build();
}@RabbitListener(queues = "dlxQueue")
public void listenDlxQueue(Message message, Channel channel) throws Exception {System.out.printf("%tc 死信队列接收到消息: %s%n", new Date(), new String(message.getBody()));
}

3.3 场景应用

  • 用户注册后发送激活邮件。
  • 订单系统中未支付订单的自动取消。
  • 退款请求处理后的自动退款。

4. 事务处理

4.1 事务的基本概念

RabbitMQ支持事务处理,允许开发者确保消息的发送和接收是原子操作。这意味着要么全部完成,要么全部失败,从而保持数据的一致性。

4.2 配置与使用

配置事务管理器并开启事务支持:

@Configuration
public class TransactionConfig {@Beanpublic RabbitTransactionManager transactionManager(CachingConnectionFactory connectionFactory) {return new RabbitTransactionManager(connectionFactory);}@Beanpublic RabbitTemplate rabbitTemplate(CachingConnectionFactory connectionFactory) {RabbitTemplate template = new RabbitTemplate(connectionFactory);template.setChannelTransacted(true);return template;}
}

4.3 生产者示例

@Transactional
@RequestMapping("/send")
public String send() {rabbitTemplate.convertAndSend("", "transQueue", "trans test 1...");int a = 5 / 0; // 故意引发异常rabbitTemplate.convertAndSend("", "transQueue", "trans test 2...");return "发送成功";
}

如果启用了事务,上面的代码将会回滚整个操作,确保没有消息被发送出去。


5. 消息分发

5.1 分发机制

当一个队列有多个消费者时,RabbitMQ会根据一定的规则将消息分配给不同的消费者。默认情况下,采用轮询方式分发,即每个消费者轮流获得一条消息。这种方法可能导致某些快速消费者空闲,而慢速消费者积压大量消息的问题。

5.2 限流与负载均衡

  • 限流:通过设置basicQos参数来控制单个消费者同时处理的消息数量,防止过载。
  • 负载均衡:设置prefetchCount=1,使得RabbitMQ每次只向一个消费者发送一条消息,在收到确认之前不会发送新的消息,从而实现更公平的负载分配。
# application.yml
spring:rabbitmq:listener:simple:acknowledge-mode: manualprefetch: 1

5.3 示例代码

@Component
public class QosQueueListener {@RabbitListener(queues = "qosQueue")public void listenQosQueue(Message message, Channel channel) throws Exception {long deliveryTag = message.getMessageProperties().getDeliveryTag();System.out.printf("接收到消息: %s, deliveryTag: %d%n", new String(message.getBody()), deliveryTag);Thread.sleep(100); // 模拟耗时操作channel.basicAck(deliveryTag, false);}
}

总结

  1. 消息确认机制

    • 通过autoAck参数控制消息的确认方式。
    • 自动确认模式适合对消息可靠性要求不高的场景;手动确认模式适用于需要高可靠性的场景。
    • 手动确认允许消费者显式地向RabbitMQ发送确认信号,确保消息被成功处理后才从队列中移除。
  2. 死信队列(Dead Letter Exchange, DLX)

    • 死信是指那些无法被正常消费的消息,可能由于消息过期、被拒绝或队列满载等原因产生。
    • 死信队列可以用来存储这些无法处理的消息,提供重试、丢弃或日志记录等功能。
    • 使用示例展示了如何配置普通队列与DLX绑定,并设置消息TTL和路由键来实现消息转移至DLX。
  3. 延迟队列

    • 延迟队列用于在指定时间之后才将消息传递给消费者。
    • 通过结合TTL与DLX,或者使用官方提供的延迟插件,可以模拟实现延迟队列功能。
    • 这种机制常应用于定时任务、订单超时处理等场景。
  4. 事务处理

    • RabbitMQ支持事务操作,确保消息的发送和接收是原子性的。
    • 通过配置事务管理器并开启事务支持,可以在发生异常时回滚整个事务,保持数据一致性。
    • 示例代码演示了如何在Spring应用中配置并使用事务管理。
  5. 消息分发策略

    • 当多个消费者订阅同一个队列时,RabbitMQ会根据一定的规则分配消息。
    • 默认采用轮询分发,但可以通过设置basicQos参数进行限流,以及调整prefetchCount实现更公平的负载均衡。
    • 适当配置有助于避免某些消费者过载而其他消费者空闲的问题,提高系统的整体吞吐量和稳定性。

以上就是 RabbitMQ 的部分高级特性,有问题可在评论区讨论,感谢阅览!!

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

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

相关文章

【Java 22 | 9】 深入解析Java 22 :Foreign Function Memory API 的改进

Java 22 对 Foreign Function & Memory API&#xff08;FFI&#xff0c;外部函数和内存 API&#xff09;进行了重要改进&#xff0c;旨在增强 Java 与本地代码及内存的交互能力。这一特性使 Java 程序能够更方便地调用非 Java 代码&#xff0c;如 C/C 库&#xff0c;同时提…

振弦式渗压计压力计算出现负值是什么原因?

振弦式渗压计作为一种高精度的测量仪器&#xff0c;被广泛应用于地质工程、水利水电工程等领域&#xff0c;用于监测土壤或结构物内部的渗水压力。然而&#xff0c;在实际应用中&#xff0c;有时会出现压力计算结果为负值的情况&#xff0c;这不仅影响数据的准确性&#xff0c;…

基于Java微信小程序的水果销售系统详细设计和实现(源码+lw+部署文档+讲解等)

详细视频演示 请联系我获取更详细的演示视频 项目运行截图 技术框架 后端采用SpringBoot框架 Spring Boot 是一个用于快速开发基于 Spring 框架的应用程序的开源框架。它采用约定大于配置的理念&#xff0c;提供了一套默认的配置&#xff0c;让开发者可以更专注于业务逻辑而不…

iLogtail 开源两周年:UC 工程师分享日志查询服务建设实践案例

作者&#xff1a;UC 浏览器后端工程师&#xff0c;梁若羽 传统 ELK 方案 众所周知&#xff0c;ELK 中的 E 指的是 ElasticSearch&#xff0c;L 指的是 Logstash&#xff0c;K 指的是 Kibana。Logstash 是功能强大的数据处理管道&#xff0c;提供了复杂的数据转换、过滤和丰富…

快充协议有哪些,都有哪些特点

什么是PD协议 PD协议是一种充电协议&#xff0c;全称为“USB Power Delivery&#xff08;USB PD&#xff09;”&#xff0c;是由USB-IF&#xff08;USB Implementers Forum&#xff09;组织制定的一种标准协议‌。它是一种基于USB接口的快速充电技术&#xff0c;可以实现高达1…

领导满意的可视化数据分析图表,原来一键配置就可以完成

数据分析图表是数据可视化的一种形式&#xff0c;它是将数据以图表的形式呈现出来&#xff0c;从而帮助人们更直观地理解数据和数据之间的关系。数据分析图表可以包括各种类型的图表&#xff0c;例如线图、柱状图、散点图、饼图等。这些图表可以用于描述单个变量的分布&#xf…

2010年国赛高教杯数学建模C题输油管的布置解题全过程文档及程序

2010年国赛高教杯数学建模 C题 输油管的布置 某油田计划在铁路线一侧建造两家炼油厂&#xff0c;同时在铁路线上增建一个车站&#xff0c;用来运送成品油。由于这种模式具有一定的普遍性&#xff0c;油田设计院希望建立管线建设费用最省的一般数学模型与方法。   1. 针对两炼…

外包干了3周,技术退步太明显了。。。。。

先说一下自己的情况&#xff0c;大专生&#xff0c;21年通过校招进入武汉某软件公司&#xff0c;干了差不多3个星期的功能测试&#xff0c;那年国庆&#xff0c;感觉自己不能够在这样下去了&#xff0c;长时间呆在一个舒适的环境会让一个人堕落!而我才在一个外包企业干了3周的功…

推荐一款流量录制回放工具:JVM-sandbox-repeater!

在软件开发和测试过程中&#xff0c;我们经常会遇到需要对网络请求进行录制和回放的需求&#xff0c;以便进行调试、测试和分析。为了模拟真实的用户请求&#xff0c;我们通常会使用各种流量录制回放工具来记录并重放网络请求。 其中&#xff0c;jvm-sandbox-repeater 是一款功…

电子电气架构 --- 智能网联汽车未来是什么样子?

我是穿拖鞋的汉子,魔都中坚持长期主义的汽车电子工程师。 老规矩,分享一段喜欢的文字,避免自己成为高知识低文化的工程师: 屏蔽力是信息过载时代一个人的特殊竞争力,任何消耗你的人和事,多看一眼都是你的不对。非必要不费力证明自己,无利益不试图说服别人,是精神上的节…

基于SpringBoot+Vue+uniapp微信小程序的婚庆摄影小程序的详细设计和实现(源码+lw+部署文档+讲解等)

项目运行截图 技术框架 后端采用SpringBoot框架 Spring Boot 是一个用于快速开发基于 Spring 框架的应用程序的开源框架。它采用约定大于配置的理念&#xff0c;提供了一套默认的配置&#xff0c;让开发者可以更专注于业务逻辑而不是配置文件。Spring Boot 通过自动化配置和约…

GO语言指针有那些限制

GO语言指针有那些限制 GO 语言的指针 一个指针变量本身存会计的只是一个内存地址 一个内存地邗在32位系统上占4个字节&#xff0c;在64位系统上占8个字节 内存地址一般用整数的16进制来表示 当一个变量声明的时候&#xff0c;GO运行时将此变量开辟一段内存&#xff0c;此内存…

遥感技术助力生态系统碳储量、碳收支、碳循环等多领域监测与模拟:森林碳储量,城市扩张,夜间灯光数据,陆地生态系统,大气温室气体监测等

目录 专题一 双碳视角下遥感技术的研究方向 专题二 生态系统碳库的遥感估算—以森林碳储量为例 专题三 生态系统碳收支的遥感模拟—以京津冀地区为例 专题四 土地利用变化碳排放效应的遥感监测—以城市扩张为例 专题五 区域能源消耗碳排放空间格局模拟—基于夜间灯光数据 …

为什么你总碰到渣男?伯克森悖论

内容预告 为什么有些女生总觉得自己总是遇到渣男&#xff1f;难道是我具备了“吸引渣男的体质”?&#xff0c;还是“好男人都绝了吗?"。今天&#xff0c;我们通过因果推断中的伯克森悖论&#xff0c;结合心理学中的认知偏差和选择偏差&#xff0c;来解析这个令人困惑的…

【word】页眉横线无法取消

小伙伴们日常想在页眉里加横线&#xff0c;直接双击页眉&#xff0c;然后在页眉横线里选择自己喜欢的横线样式就可以了。 但今天我遇到的这个比较奇特&#xff0c;有些页有这个横线&#xff0c;有些页没有&#xff0c;就很奇怪。 最后排查完&#xff0c;发现是只有标题2的页…

15分钟学Go 第4天:Go的基本语法

第4天&#xff1a;基本语法 在这一部分&#xff0c;将讨论Go语言的基本语法&#xff0c;了解其程序结构和基础语句。这将为我们后续的学习打下坚实的基础。 1. Go语言程序结构 Go语言程序的结构相对简单&#xff0c;主要包括&#xff1a; 包声明导入语句函数语句 1.1 包声…

5、JavaScript(三)

20.this对象 对于要绑定的多个对象的事件内容相同时可以使用循环来绑定&#xff0c;注意这时要使用this对象拿到当前调用函数的对象的属性和方法&#xff0c;不能直接使用循环变量作为角标。 1 this 对象基础内容 <!-- 大坑坑坑坑&#xff01;&#xff01;&#xff01;&am…

arm架构ceph pacific部署

背景 合作伙伴实验室的华为私有云原来使用单点的nfs做为存储设备&#xff0c;现有两方面考量&#xff0c;业务需要使用oss了&#xff0c;k8s集群及其他机器也需要一套可扩展的分布式文件系统 部署ceph 初始机器配置规划 IP配置主机名Role10.17.3.144c8g1T数据盘ceph-node01…

录屏不再难!四款免费电脑桌面录屏工具测评报告

作为一个办公室文员&#xff0c;我经常需要录制一些操作演示视频或者会议记录。最近&#xff0c;我尝试了四款免费的电脑桌面录屏工具&#xff1a;福昕录屏大师、转转大师录屏、爱拍录屏和Screen Studio。今天就来跟大家分享一下我的使用体验&#xff0c;希望能帮到和我有同样需…

vue综合指南(六)

​&#x1f308;个人主页&#xff1a;前端青山 &#x1f525;系列专栏&#xff1a;Vue篇 &#x1f516;人终将被年少不可得之物困其一生 依旧青山,本期给大家带来Vuet篇专栏内容:vue综合指南 目录 101、Vue 框架怎么实现对象和数组的监听&#xff1f; 102、Proxy 与 Object.d…