RabbitMq 入门实例详解+实例代码
RabbitMq 入门实例详解+实例代码
最近有项目又有使用RabbitMQ,使用过程中看到有使用 “AmqpAdmin” 后进行详细研究为什么会用 创建 Queue、Exchange 还用AmqpAdmin.delcareQueue,经过深入的查资料和思考发现 其实没必要 在@Configuration 类中 Return new Queue()中使用; 经过这次遇到的问题也正好对RabbitMq整体梳理,下面是干货满满的说明和实例代码。
RabbitMQ 按装请自已查找,这里不在赘述。
1、 实例中有生产者:
confirmpusher:发送方
confirmreceive:接收方
实例confirmpusher 详细搭建
1)Pom 文件节点:
<?xml version="1.0" encoding="UTF-8"?> <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd"> <modelVersion>4.0.0</modelVersion> <parent> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-parent</artifactId> <version>2.6.3</version> <relativePath/> <!-- lookup parent from repository --> </parent> <groupId>com.liyanbomq</groupId> <artifactId>confirmpusher</artifactId> <version>0.0.1-SNAPSHOT</version> <name>confirmpusher</name> <description>Demo project for Spring Boot</description> <properties> <java.version>1.8</java.version> </properties> <dependencies> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-amqp</artifactId> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> </dependency> <dependency> <groupId>org.projectlombok</groupId> <artifactId>lombok</artifactId> <optional>true</optional> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-test</artifactId> <scope>test</scope> </dependency> <dependency> <groupId>org.springframework.amqp</groupId> <artifactId>spring-rabbit-test</artifactId> <scope>test</scope> </dependency> </dependencies> <build> <plugins> <plugin> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-maven-plugin</artifactId> <configuration> <excludes> <exclude> <groupId>org.projectlombok</groupId> <artifactId>lombok</artifactId> </exclude> </excludes> </configuration> </plugin> </plugins> </build> </project>
2)yml文件
spring:
rabbitmq:
host: 192.168.88.129
username: liyanbo
password: liyanbo
#开启发送确认机制,将来消息到达交换机以后有一个回调
publisher-confirm-type: correlated
#消息到达消息队列回调(如果消息没有成功到达队列,会触发回调方法)
publisher-returns: true
template:
retry:
enabled: true # 开启重发机制
initial-interval: 1000ms #间隔 1秒
max-attempts: 6 #最多发6次
multiplier: 1.2 #每次间隔 时间*1.2
max-interval: 10000ms #每次最大间隔时间
port: 5672
listener:
simple:
acknowledge-mode: manual
server:
port: 7004
3)读取RabbitMQ相关配置后声明 队列、交换机,进行交换机、队列绑定
@Configuration
RabbitConfigurationSure
package com.liyanbomq.confirmpusher.configuration; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.amqp.core.*; import org.springframework.amqp.rabbit.connection.CorrelationData; import org.springframework.amqp.rabbit.core.RabbitTemplate; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import javax.annotation.PostConstruct; /** * 用于mq消息确认 * * 总结 * 一、确认消息发送成功到交换机和消息队列 * 1、调用回调方法 * 利用 yml文件中参数进行设置发送失败消息进行重新发送 * template: * retry: * enabled: true # 开启重发机制 * initial-interval: 1000ms #间隔 1秒 * max-attempts: 6 #最多发6次 * multiplier: 1.2 #每次间隔 时间*1.2 * max-interval: 10000ms #每次最大间隔时间 * * * 二、 确认消息发送成功到交换机和消息队列 调用回调方法 * 1、在回调方法中 记录发送失败的数据记录 (如用mysql存储) * 2、制作定时任务处理未成功发送的消息 * */ @Configuration public class RabbitConfigurationSure implements RabbitTemplate.ConfirmCallback,RabbitTemplate.ReturnsCallback { public static final String CONFIRM_QUEUE_NAME="confirm_queue"; public static final String EXCHANGE_NAME="confirm_exchange"; public final Logger logger= LoggerFactory.getLogger(RabbitConfigurationSure.class); @Autowired RabbitTemplate rabbitTemplate; /** * 这样设置 setReturnsCallback、setConfirmCallback才会起作用 */ @PostConstruct public void initRabbitMq(){ rabbitTemplate.setReturnsCallback(this); rabbitTemplate.setConfirmCallback(this); } /** * 创建队列 * @return */ @Bean public Queue getConfirmQueue(){ /** * parameter 1 队列名称 * durable:持久化 * exclusive: 排他其它连接也可操作 * autodelete:没有消费连接时不会自动删除 */ return new Queue(CONFIRM_QUEUE_NAME,true,false,false); } /** * 交换机 * @return */ @Bean public DirectExchange getConirmExchange(){ return new DirectExchange(EXCHANGE_NAME,true,false); } /** * 绑定消息队列到交换机上 * @return */ @Bean public Binding getConfirmBinding(){ return BindingBuilder.bind(getConfirmQueue()).to(getConirmExchange()).with(CONFIRM_QUEUE_NAME); } /** * 消息成功到达交换机会触发回调该方法 * @param correlationData * @param ack 是否成功到达交换机 * @param cause 如果未成功到达,原因 */ @Override public void confirm(CorrelationData correlationData, boolean ack, String cause) { if(ack){ System.out.println("成功到达交换机"); logger.info("{}成功到达交换机",correlationData.getId()); }else{ System.out.println("未成功到达交换机,原因:"+cause); logger.info("{}未成功到达交换机,原因:{}",correlationData.getId(),cause); } } /** * 消息未成功到达该队列会触发回调该方法 * @param returnedMessage */ @Override public void returnedMessage(ReturnedMessage returnedMessage) { System.out.println("消息成功到达队列"); logger.info("{}消息成功到达队列",returnedMessage.getMessage().getMessageProperties().getMessageId()); } }
4) 发起调用 往消息队列写信息
package com.liyanbomq.confirmpusher.controller; import com.liyanbomq.confirmpusher.configuration.RabbitConfigurationSure; import org.springframework.amqp.rabbit.connection.CorrelationData; import org.springframework.amqp.rabbit.core.RabbitTemplate; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.RequestParam; import org.springframework.web.bind.annotation.RestController; import java.util.UUID; @RestController public class pushmqController { @Autowired RabbitTemplate rabbitTemplate; @RequestMapping("send") public String sendmqInfo(@RequestParam("msg") String msg){ // try { rabbitTemplate.convertAndSend(RabbitConfigurationSure.EXCHANGE_NAME,RabbitConfigurationSure.CONFIRM_QUEUE_NAME,msg,new CorrelationData(UUID.randomUUID().toString())); // } catch (Exception e) { // e.printStackTrace(); // } return "it is msg:"+msg; } @RequestMapping("sendfanout") public String sendmqInfofanout(@RequestParam("msg") String msg){ // try { rabbitTemplate.convertAndSend(RabbitConfigurationSure.EXCHANGE_NAME+"fanout",RabbitConfigurationSure.CONFIRM_QUEUE_NAME+"fanout",msg,new CorrelationData(UUID.randomUUID().toString())); // } catch (Exception e) { // e.printStackTrace(); // } return "it is msg:"+msg; } }
4)Springboot 启动项,运行
package com.liyanbomq.confirmpusher; import org.springframework.boot.SpringApplication; import org.springframework.boot.autoconfigure.SpringBootApplication; @SpringBootApplication public class ConfirmpusherApplication { public static void main(String[] args) { SpringApplication.run(ConfirmpusherApplication.class, args); } }
5)通过Controller发起调用
http://localhost:7004/send?msg=1234
2、接收都 代码搭建 1)pom文件、2)yml文件、3)配置信息读取与 发送者搭建一样
1、
2、
3、
4、接收者代码
package com.liyanbomq.confirmreceive.receive; import com.liyanbomq.confirmreceive.configuration.RabbitConfigurationSure; import com.rabbitmq.client.Channel; import org.springframework.amqp.core.Message; import org.springframework.amqp.rabbit.annotation.RabbitListener; import org.springframework.stereotype.Component; import java.io.IOException; @Component public class ReceiverService { @RabbitListener(queues = RabbitConfigurationSure.CONFIRM_QUEUE_NAME) public void receiveMsg(Message message, Channel channel){ // 消息标识 long deliveryTag= message.getMessageProperties().getDeliveryTag(); try { byte [] bytes=message.getBody(); String mess=new String(bytes); System.out.println("mess = " + mess); // int l=2/2/0; // 手动确认签收 第一个参数是消息标记 第二个参数fasle 只确认当前消息,true表示之前所有的消息都确认成功 channel.basicAck(deliveryTag,false); } catch (Exception e) { try { // 标示签收失败,再次放入队列中 第三个参烽 requeue 再次放入队列 channel.basicNack(deliveryTag,false,true); } catch (IOException ioException) { ioException.printStackTrace(); } e.printStackTrace(); } } }
二、关于自定义配置RabbitMq 和AmqpAdmin.declare 说明
一、关于 rabbitMq配置是否需要显式写代码
/**
*
*组装配置 rabbitMq - connectionFactory
* 1、如果默认配置节点设置 不用该组装connectionFactory
* spring:
* rabbitmq:
* username: liyanbo
* password: liyanbo
* host: 192.168.88.129
* port: 5672
*2、如果 自定义配置像下面 自定义pos:就需要 组装配置 connectionFactory
* spring:
* * rabbitmq:
* pos:
* * username: liyanbo
* * password: liyanbo
* * host: 192.168.88.129
* * port: 5672
*
* @return
*/
@Bean(name = "connectionFactory")
public CachingConnectionFactory connectionFactory(){
CachingConnectionFactory connectionFactory = new CachingConnectionFactory();
connectionFactory.setHost("192.168.88.129");
connectionFactory.setPort(5672);
connectionFactory.setUsername("liyanbo");
connectionFactory.setPassword("liyanbo");
connectionFactory.setPublisherReturns(true);
return connectionFactory;
}
@Bean(name = "rabbitTemplate")
public RabbitTemplate rabbitTemplate(@Qualifier("connectionFactory") CachingConnectionFactory connectionFactory) {
RabbitTemplate rabbitTemplate = new RabbitTemplate(connectionFactory);
return rabbitTemplate;
}
@Bean(name = "amqpAdmin")
public AmqpAdmin amqpAdmin(@Qualifier("connectionFactory") CachingConnectionFactory connectionFactory){
AmqpAdmin amqpAdmin=new RabbitAdmin(connectionFactory);
return amqpAdmin;
}
二、 AmqpAdmin amqpAdmin 是否需要
1、使用了下面 @Configuration 就不需要使用
@Configuration
public class RabbitConfigurationSure {
2、在下面过程中 会自动生成声明 Queue(源码本有逻辑), 不需要使用amqpAdmin.declareQueue(queue); 只有使用测试或普通类不使用 @Configuration 才需要amqpAdmin.declareQueue(queue)
@Bean
public Queue supplyRabbitQueue() {
return new Queue("SupplyQueue");
}
ps:AmqpAdmin用于创建/删除 Exchange、Queue、Binding 和初始化RabbitMQ
AmqpAdminn 说明地址:Spring Boot - RabbitMQ源码分析 - 知乎 (zhihu.com)
总结:
为消息安全确认消费有2种方式
1、为此AMQP协议在建立之初就考虑到这种情况而提供了事务机制。
使用事务机制的话会“吸干”RabbitMQ的性能,那么有没有更好的方法既能保证消息发送方确认消息已经正确送达,又能基本上不带来性能上的损失呢? 从AMQP协议层面来看并没有更好的办法,但是RabbitMQ提供了一个改进方案,即发送方确认机制(publisher confirm)
2、发送方确认机制(publisher confirm)
RabbitMQ消息队列:ACK机制
源码下载地址:https://files.cnblogs.com/files/liyanbofly/confrimmqinfo.rar
如有疑问或问题可以沟通讨论qq:626382542