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
posted @ 2022-07-03 11:50  xiaoBai1001  阅读(2070)  评论(0编辑  收藏  举报