SpringBoot mq快速上手

1.依赖

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-amqp</artifactId>
</dependency>

2.示例代码

基础信息配置

java 复制代码
package com.example.demo.config;

import org.springframework.amqp.core.*;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

@Configuration
public class RabbitConfig {
    public static final String EXCHANGE = "boot-EXCHANGE";

    public static final String QUEUE = "boot-queue";

    public static final String routingKey = "*.black.*";
    @Bean
    public Queue bootQueue(){
        return QueueBuilder.durable(QUEUE).build();
    }
    @Bean
    public Exchange bootExchange(){
        return ExchangeBuilder.topicExchange(EXCHANGE).build();
    }
    @Bean
    public Binding binding(Queue bootQueue,Exchange bootExchange){
        return BindingBuilder.bind(bootQueue).to(bootExchange).with(routingKey).noargs();

    }
}

实例配置代码如上

实操代码如下:

java 复制代码
  rabbitTemplate.convertAndSend(EXCHANGE,"white.black.hello","nihao");
        System.out.println("消息发送成功");

mq确保生产者发送到交换机链路正常:

基本:

java 复制代码
  //设置回调(保证消息发到交换机上)
        channel.confirmSelect();
        //设置回调函数
        channel.addConfirmListener(new ConfirmListener() {
            @Override
            public void handleAck(long deliveryTag, boolean multiple) throws IOException {
                System.out.println("消息成功发送到交换机");
            }

            @Override
            public void handleNack(long deliveryTag, boolean multiple) throws IOException {
                System.out.println("消息没有发送到交换机,请重试");
            }
        });

SpringBoot

java 复制代码
 // publisher-confirm-type: correlated  # 新版本 spring配置
    rabbitTemplate.setConfirmCallback(new RabbitTemplate.ConfirmCallback() {

            @Override
            public void confirm(CorrelationData correlationData, boolean ack, String cause) {
                if(ack){
                    System.out.println("消息成功发送到交换机");
                }
                else{
                    System.out.println("消息没有发送到交换机");
                }
            }
        });

该方法中的ack为true,或false代表如上,该方法会被异步调用,不会阻塞主方法,效率高,可以在这里记下来,记到数据库中,事后进行补充。。

交换机路由到队列正常

java 复制代码
//保证消息发到Queue
        channel.addReturnListener(new ReturnCallback() {
            @Override
            public void handle(Return returnMessage) {
                System.out.println("消息未发送到交换机,请进行相关处理");//111
            }
        });

该方法在路由队列失败时进行回调

java 复制代码
  // publisher-returns: true # 开启Return机制
 rabbitTemplate.setReturnsCallback(new RabbitTemplate.ReturnsCallback() {
            @Override
            public void returnedMessage(ReturnedMessage returned) {
                String msg = new String(returned.getMessage().getBody());
                System.out.println("msg路由队列失败,请进行补救操作"+msg);
            }
        });

该方法只会在队列路由失败时被调用,属于回调方法

设置队列持久化(不是指durable设置为true,是指mq重启后,队列依然有消息)

java 复制代码
  //设置消息持久化
        AMQP.BasicProperties prop =new AMQP.BasicProperties().builder()
                .deliveryMode(2)
                .build();
        //6.发送消息
        channel.basicPublish(Exchange,"",prop,"nihao".getBytes(StandardCharsets.UTF_8));

设置属性,并把这个属性带上,deliveryMode设置为2即可

java 复制代码
  rabbitTemplate.convertAndSend(EXCHANGE, "big.black.dog", "nihao", new MessagePostProcessor() {
            @Override
            public Message postProcessMessage(Message message) throws AmqpException {
                message.getMessageProperties().setDeliveryMode(MessageDeliveryMode.PERSISTENT);
                return message;
            }
        });

springboot中,需要用 MessagePostProcessor里的message设置deleveryMode 并且设置为枚举类型的PERSISTENT类型,并且记得返回消息,不然无法设置队列类型为持久化

保证消息被消费者正常消费

java 复制代码
  //4.监听消息
        DefaultConsumer callback=new DefaultConsumer(channel){
            @Override
            public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body)
                    throws IOException {
                System.out.println("消费者1好收到消息"+new String(body,"utf-8"));
                channel.basicAck(envelope.getDeliveryTag(),false);
            }
        };
       channel.basicConsume(QUEUE,false,callback);

mq在原始代码的时候需要在消费者端设置自动ack为false来进行处理,并在方法里进行手动ack。

java 复制代码
  listener:
      simple:
        acknowledge-mode: manual #开启手动ACK
        prefetch: 10 #消费者一次拿走10个消息

配置如上

@Component
public class Consumer {
    @RabbitListener(queues = RabbitConfig.QUEUE)
    public void Consume(String msg, Channel channel, Message message) throws IOException {
        System.out.println("msg"+msg);
        System.out.println("标示符"+message.getMessageProperties().getCorrelationId());
        channel.basicAck(message.getMessageProperties().getDeliveryTag(),false);
    }
}

springboot需要在配置文件里设置值为manual,prefetch表示一次可以取走的消息

相关推荐
customer082 小时前
【开源免费】基于SpringBoot+Vue.JS体育馆管理系统(JAVA毕业设计)
java·vue.js·spring boot·后端·开源
Miketutu2 小时前
Spring MVC消息转换器
java·spring
乔冠宇2 小时前
Java手写简单Merkle树
java·区块链·merkle树
LUCIAZZZ3 小时前
简单的SQL语句的快速复习
java·数据库·sql
komo莫莫da4 小时前
寒假刷题Day19
java·开发语言
小小虫码4 小时前
项目中用的网关Gateway及SpringCloud
spring·spring cloud·gateway
计算机-秋大田4 小时前
基于微信小程序的电子竞技信息交流平台设计与实现(LW+源码+讲解)
spring boot·后端·微信小程序·小程序·课程设计
S-X-S4 小时前
算法总结-数组/字符串
java·数据结构·算法
linwq85 小时前
设计模式学习(二)
java·学习·设计模式
桦说编程5 小时前
CompletableFuture 超时功能有大坑!使用不当直接生产事故!
java·性能优化·函数式编程·并发编程