首先实现生产者发送消息和队列的持久化,这部分摘抄自http://blog.720ui.com/2017/rabbitmq_action_durable/
要从奔溃的 RabbitMQ 中恢复的消息,我们需要做消息持久化。如果消息要从 RabbitMQ 奔溃中恢复,那么必须满足三点,且三者缺一不可。
- 交换器必须是持久化。
- 队列必须是持久化的。
- 消息必须是持久化的。
原生的实现方式
原生的 RabbitMQ 客户端需要完成三个步骤。
第一步,交换器的持久化。
// 参数1 exchange :交换器名 // 参数2 type :交换器类型 // 参数3 durable :是否持久化 channel.exchangeDeclare(EXCHANGE_NAME, "topic", true);
第二步,队列的持久化。
// 参数1 queue :队列名 // 参数2 durable :是否持久化 // 参数3 exclusive :仅创建者可以使用的私有队列,断开后自动删除 // 参数4 autoDelete : 当所有消费客户端连接断开后,是否自动删除队列 // 参数5 arguments channel.queueDeclare(QUEUE_NAME, true, false, false, null);
第三步,消息的持久化。
// 参数1 exchange :交换器 // 参数2 routingKey : 路由键 // 参数3 props : 消息的其他参数,其中 MessageProperties.PERSISTENT_TEXT_PLAIN 表示持久化 // 参数4 body : 消息体 channel.basicPublish("", queue_name, MessageProperties.PERSISTENT_TEXT_PLAIN, message.getBytes());
Spring AMQP 的实现方式
Spring AMQP 是对原生的 RabbitMQ 客户端的封装。一般情况下,我们只需要定义交换器的持久化和队列的持久化。
其中,交换器的持久化配置如下。
// 参数1 name :交互器名 // 参数2 durable :是否持久化 // 参数3 autoDelete :当所有消费客户端连接断开后,是否自动删除队列 new TopicExchange(name, durable, autoDelete)
此外,还需要再配置队列的持久化。
// 参数1 name :队列名 // 参数2 durable :是否持久化 // 参数3 exclusive :仅创建者可以使用的私有队列,断开后自动删除 // 参数4 autoDelete : 当所有消费客户端连接断开后,是否自动删除队列 new Queue(name, durable, exclusive, autoDelete);
至此,RabbitMQ 的消息持久化配置完毕。
那么,消息的持久化难道不需要配置么?确实如此,我们来看下源码。
一般情况下,我们会通过这种方式发送消息。
rabbitTemplate.convertAndSend(exchange, routeKey, message);
其中,调用了 convertAndSend(String exchange, String routingKey, final Object object) 方法。
@Override public void convertAndSend(String exchange, String routingKey, final Object object) throws AmqpException { convertAndSend(exchange, routingKey, object, (CorrelationData) null); }
接着,用调用了 convertAndSend(String exchange, String routingKey, final Object object, CorrelationData correlationData) 方法。
public void convertAndSend(String exchange, String routingKey, final Object object, CorrelationData correlationData) throws AmqpException { send(exchange, routingKey, convertMessageIfNecessary(object), correlationData); }
此时,最关键的方法出现了,它是 convertMessageIfNecessary(final Object object)。
protected Message convertMessageIfNecessary(final Object object) { if (object instanceof Message) { return (Message) object; } return getRequiredMessageConverter().toMessage(object, new MessageProperties()); }
其中,关键的是 MessageProperties 类,它持久化的策略是 MessageDeliveryMode.PERSISTENT,因此它会初始化时默认消息是持久化的。
public class MessageProperties implements Serializable { public MessageProperties() { this.deliveryMode = DEFAULT_DELIVERY_MODE; this.priority = DEFAULT_PRIORITY; } static { DEFAULT_DELIVERY_MODE = MessageDeliveryMode.PERSISTENT; DEFAULT_PRIORITY = Integer.valueOf(0); } }
在消费者端,我这里采用的是默认的exchange,所以有很多配置没有使用,可以参考其他的
import com.rabbitmq.client.Channel; import lombok.extern.slf4j.Slf4j; import org.springframework.amqp.rabbit.annotation.RabbitHandler; import org.springframework.amqp.rabbit.annotation.RabbitListener; import org.springframework.amqp.support.AmqpHeaders; import org.springframework.messaging.handler.annotation.Header; import org.springframework.messaging.simp.SimpMessagingTemplate; import org.springframework.stereotype.Component; import javax.annotation.Resource; import java.io.IOException; import java.util.concurrent.CompletableFuture; import java.util.concurrent.Executor; @Component @RabbitListener(queues = "rotork_websocketQueue") @Slf4j public class WebsocketConsumer { @Resource private SimpMessagingTemplate simpMessagingTemplate; @Resource private Executor poolTaskExecutor; @RabbitHandler public void send(String msg, Channel channel, @Header(AmqpHeaders.DELIVERY_TAG) long tag){ CompletableFuture.runAsync(() ->{ try { simpMessagingTemplate.convertAndSend("/topic/msg", msg); //不需要重新发送消息 channel.basicAck(tag,false); }catch (Exception e){ try { // 消费失败,重新发送消息 channel.basicNack(tag, false, true); } catch (IOException ioe) { log.error("websocket消费重新获取消费消息错误:{}", ioe); } throw new RuntimeException(e); } }, poolTaskExecutor).exceptionally(e -> { log.error("websocket消费消息失败", e); throw new RuntimeException(e); }); } }
相关推荐
基于SpringBoot+RabbitMQ用户注册实现异步发送验证码源码。基于SpringBoot+RabbitMQ用户注册实现异步发送验证码源码。基于SpringBoot+RabbitMQ用户注册实现异步发送验证码源码。...基于SpringBoot+Rabbit
在"springboot-rabbit"这个项目中,我们可以预期找到以下文件结构和内容: 1. `pom.xml`:包含Spring Boot、Spring AMQP和HikariCP相关的依赖。 2. `application.properties`/`application.yml`:配置RabbitMQ...
springboot整合rabbit+redis+mysql实现用户购买商品时提交订单后的过程; 用户提交后,生成订单,用户有XX分钟时间完成支付 如果用户XX分钟内完成支付,则更新订单为已支付,仓库扣库存,发送短信 如果用户XX分钟后...
通过整合这两者,我们可以利用消息队列的优势,实现应用之间的异步通信和解耦。 首先,要开始这个整合过程,你需要确保已经安装并运行了RabbitMQ服务器。可以从官方网站下载适合你操作系统的版本,并按照官方文档的...
springboot Rabbit死信队列实现,rocketMq重试消息实现 基于springboot2.15版本,最新rabbit和rocktMq 中间件实例,亲测可用
rabbitmq+websocket(SpringBoot版)实现分布式消息推送 本来想用websocket做一个消息推送 可是分布式环境下不支持session共享因为服务器不同 所以采用 rabbitMQ+webSocket实现分布式消息推送 生产者将消息 发送给 ...
在本项目"RabbitMQ实战-多线程-springboot-rabbit.zip"中,我们将深入探讨如何使用RabbitMQ作为消息中间件,结合Spring Boot框架和多线程技术来实现高效、可扩展的分布式系统。Spring Boot简化了RabbitMQ的集成,而...
rabbit mq windows 64 较新,很实用,消息中间件服务端。
在IT行业中,消息队列(MQ)是一种常用于应用程序间通信的技术,它允许不同系统之间异步传递消息,从而解耦各个组件,提高系统的可扩展性和可靠性。RabbitMQ是其中一个广泛使用的开源消息代理和队列服务器,它实现了...
本篇文章将围绕"简单封装spring-rabbit实现mq组件化"这一主题,探讨如何通过Spring框架和RabbitMQ来构建可复用的MQ组件。 首先,我们需要了解Spring框架的Spring AMQP模块,这是Spring对RabbitMQ的支持。它提供了...
我们将会从基础概念出发,逐步深入到具体实践,包括如何配置 Spring Boot 项目以连接和操作 RabbitMQ,如何创建和管理消息队列、发布和订阅消息,以及处理消息确认和异常等高级特性。 #### 二、RabbitMQ 基础知识 ...
springboot集成rabbitmq的简单使用,介绍了springboot集成rabbitmq的使用,利用的交换机、队列、路由key来实现的例子
spring boot 集成rabbit mq 成功demo,spring boot 集成rabbit mq 成功demo
RabbitMQ连接池+SpringBoot实现。通过连接池实现将高效的管理RabbitMQ的Connection,并与springboot进行整合,实现消息发送,获取队列列表等功能。基于此可以进行更多功能的扩充。
rabbit mq server 常用命令 rabbit mq server 常用命令 rabbit mq server 常用命令
在这个“rabbit mq的demo更新”中,我们将探讨如何通过编程实现与RabbitMQ的连接,发送和接收消息,以及如何配置消息的持久化和客户端订阅。 首先,连接RabbitMQ通常需要一个客户端库,如Java的`rabbitmq-amqp-...
3. 持久化:对于重要的消息,可以设置队列和消息的持久化,保证在服务器重启后不丢失数据。 4. 高可用性:通过RabbitMQ集群提高服务的可用性和容错性。 总的来说,RabbitMQ是一个强大的消息中间件,它允许应用程序...
docker 安裝 rabbit mq 並測試 http://knight-black-bob.iteye.com/blog/2395713
本项目基于SpringBoot框架,结合Redis、Zookeeper和RabbitMQ实现了分布式锁,让我们一起深入探讨这三个组件在分布式锁中的作用和实现机制。 **SpringBoot** 是一个轻量级的Java开发框架,它简化了新Spring应用的...
本资料包"rabbit-mq-ack-direct-consumer.zip"包含的是关于RabbitMQ中消费者实现的代码示例,特别关注了消息确认(Message Acknowledgement)和Direct交换机模式。 首先,我们来理解RabbitMQ中的消息确认机制。在...