一、消息队列MQ的三大功能

1.异步处理

2.应用解耦:A系统的代码需要调用B系统的接口才能完成一个任务,假设B两个系统崩溃了,那么A系统就会出现超时,导致整个系统不可用。如果使用消息队列可以将A调用B接口的参数放到队列里面,A代码里面不再直接去调用B系统的接口,而是让B系统订阅消费A发送过来的的消息(参数),B系统根据这些消息(参数)修改自己的逻辑即可,B系统维护时,不会影响A系统的使用。

3.流量控制:系统最高能承受100万并发,超出了就用队列进行排队。

二、高级消息队列交换机模型:

      1. P2P(点对点)-----     direct exchange    路由键精确匹配连接队列

      2.pub/sub (发布/订阅)  以路由键匹配规则不同,划分出不同类型

              |-----  fanout exchange  没有路由键概念,扇出交换机直连绑定的队列

topic exchange

              |------  headers exchange

              |------  system exchange

补充:

消息可靠投递:要消息要不丢失,保证三个阶段不出现问题,如图:

消息队列msgsnd是什么缩写 消息队列功能_sql

方案1:结合Rabbitmq 确认回调和回退回调处理

消息队列msgsnd是什么缩写 消息队列功能_消息队列msgsnd是什么缩写_02

   <<<<<<<<注意:发送消息前要将消息保存到Mysql起来>>>>>>>>

package com.atguigu.rabbitmq.springbootrabbitmq.config;

import lombok.extern.slf4j.Slf4j;
import org.springframework.amqp.core.Message;
import org.springframework.amqp.rabbit.connection.CorrelationData;
import org.springframework.amqp.rabbit.core.RabbitTemplate;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.lang.Nullable;
import org.springframework.stereotype.Component;
import org.springframework.util.StringUtils;

import javax.annotation.PostConstruct;

/**
 * User: ldj
 * Date: 2023/3/13
 * Time: 20:19
 * Description: No Description
 */
@Slf4j
@Component
public class RabbitmqCallbackConfig {

    @Autowired
    private RabbitTemplate rabbitTemplate;
    
    //第二种写法(推荐) 
    @PostConstruct
    public void initRabbitTemplate() {

        rabbitTemplate.setConfirmCallback((@Nullable CorrelationData correlationData, boolean ack, @Nullable String cause) -> {

            if (!ack) {
                //TODO 消息没成功抵达Broker,修改Mysql消息状态标识,并从缓存获取再次发送
            }
        });


        rabbitTemplate.setReturnCallback((Message message, int replyCode, String replyText, String exchange, String routingKey) -> {

            if (!StringUtils.isEmpty(replyText)) {
                //TODO 消息没成功抵达Queue,修改Mysql消息状态标识,弄一个定时任务读取数据mysql并向Rabbitmq发送消息
            }
        });
    }

}

方案2:备用交换机也能很好解决第二阶段问题

消息队列msgsnd是什么缩写 消息队列功能_消息队列msgsnd是什么缩写_03

 三、docker 安装rabbitmq

docker run -d --name rabbitmq -p 5671:5671 -p 5672:5672 -p 4369:4369 -p 25672:25672 -p 15671:15671 -p 15672:15672 rabbitmq:management

#5671、5672 (AMQP端口)
#15672(web可视管理后台端口)
#4369、25672(Erlang发现&集群端口)
#61613、61614(STOMP协议端口)
#1883、8883(MQTT协议端口)


#开机自启
docker update rabbitmq --restart=always

四、springBoot整合rabbitmq 以及测试

#1.引入rabbitmq启动依赖

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


#2.简单配置文件application.properties,必须以spring.rabbitmq开头

     spring.rabbitmq.host=192.168.xxx.xxx
     spring.rabbitmq.virtual-host=/
     spring.rabbitmq.port=5672
     #默认账号密码
     spring.rabbitmq.username=guest
     spring.rabbitmq.password=guest


#3.启动类加上@EnableRabbit

 配置文件 application.properties

#rabbitmq安装在虚拟机的地址
spring.rabbitmq.host=192.168.126.131
#rabbitmq又进行虚拟服务器划分,默认是/
spring.rabbitmq.virtual-host=/
#默认端口
spring.rabbitmq.port=5672
#默认账号密码
spring.rabbitmq.username=guest
spring.rabbitmq.password=guest

#开启消息确认回调(PB) 
spring.rabbitmq.publisher-confirm-type=correlated
#确认消息传达队列
spring.rabbitmq.publisher-returns=true
#异步发送优先回调返回确认
spring.rabbitmq.template.mandatory=true
#手动应答ack,自动模式宕机会导致消息丢失
spring.rabbitmq.listener.simple.acknowledge-mode=manual
#限制消费者每秒从队列拉取的消息的数量
spring.rabbitmq.listener.simple.prefetch=1000

 1.组件准备

package com.atguigu.rabbitmq.springbootrabbitmq.config;

import com.atguigu.rabbitmq.springbootrabbitmq.config.properties.RabbitmqProperties;
import org.springframework.amqp.core.*;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

import java.util.HashMap;

/**
 * User: ldj
 * Date: 2023/3/13
 * Time: 17:56
 * Description: No Description
 */
@Configuration
public class PublishConfirmConfig {

    public static final String CONFIRM_EXCHANGE_NAME = "CF-EXC";
    public static final String BACKUP_EXCHANGE_NAME = "BU-EXC";

    public static final String CONFIRM_QUEUE_NAME = "CF-QUE";
    public static final String BACKUP_QUEUE_NAME = "BU-QUE";
    public static final String WARNING_QUEUE_NAME = "WAR-QUE";

    /*
     *  P ----> CF-EXC --(b)--> CF-QUE ----> confirmConsumer                            |
     *            |                               ^
     *            v                               |
     *          BU-EXC ----> BU-QUE----------------
     *            |
     *            v
     *          WAR-QUE -------------------> warningConsumer
     */

    /**
     * 声明队列
     */
    @Bean("confirmQueue")
    public Queue confirmQueue() {
        return QueueBuilder.durable(CONFIRM_QUEUE_NAME).build();
    }

    @Bean("backupQueue")
    public Queue backupQueue() {
        return QueueBuilder.durable(BACKUP_QUEUE_NAME).build();
    }

    @Bean("warningQueue")
    public Queue warningQueue() {
        return QueueBuilder.durable(WARNING_QUEUE_NAME).build();
    }

    /**
     * 声明交换机
     */
    @Bean("confirmExchange")
    public TopicExchange confirmExchange() {
        HashMap<String, Object> arguments = new HashMap<>();

        return ExchangeBuilder
                .topicExchange(CONFIRM_EXCHANGE_NAME)
                .durable(true)
                .alternate(BACKUP_EXCHANGE_NAME) //备份交换机
                .withArguments(arguments)
                .build();
    }

    @Bean("backupExchange")
    public FanoutExchange backupExchange() {
        return new FanoutExchange(BACKUP_EXCHANGE_NAME, true, false, new HashMap<>());
    }

    /**
     * 交换机与队列捆绑
     */
    @Bean
    public Binding confirmQueueToConfirmExchange(@Qualifier("confirmQueue") Queue confirmQueue,
                                                 @Qualifier("confirmExchange") TopicExchange confirmExchange,
                                                 RabbitmqProperties properties) {
        return BindingBuilder.bind(confirmQueue).to(confirmExchange).with(properties.getConfirmBingingKey());
    }

    @Bean
    public Binding backupQueueToBackupExchange(@Qualifier("backupQueue") Queue confirmQueue,
                                               @Qualifier("backupExchange") FanoutExchange confirmExchange) {
        return BindingBuilder.bind(confirmQueue).to(confirmExchange);
    }

    @Bean
    public Binding warningQueueToBackupExchange(@Qualifier("warningQueue") Queue confirmQueue,
                                                @Qualifier("backupExchange") FanoutExchange confirmExchange) {
        return BindingBuilder.bind(confirmQueue).to(confirmExchange);
    }

}

 2.消费者1

package com.atguigu.rabbitmq.springbootrabbitmq.listener;

import com.atguigu.rabbitmq.springbootrabbitmq.config.PublishConfirmConfig;
import com.rabbitmq.client.Channel;
import lombok.extern.slf4j.Slf4j;
import org.springframework.amqp.core.Message;
import org.springframework.amqp.rabbit.annotation.RabbitHandler;
import org.springframework.amqp.rabbit.annotation.RabbitListener;
import org.springframework.stereotype.Component;

/**
 * User: ldj
 * Date: 2023/3/13
 * Time: 18:45
 * Description: No Description
 */
@Slf4j
@Component
@RabbitListener(queues = {PublishConfirmConfig.CONFIRM_QUEUE_NAME,PublishConfirmConfig.BACKUP_QUEUE_NAME})
public class ConfirmListener {

    @RabbitHandler
    public void receiveConfirmMsg(Message message, Channel channel, String msg) {

        log.info("接收到队列CF-QUE/BU-QUE消息:{}", msg);
    }
}

消费者2

package com.atguigu.rabbitmq.springbootrabbitmq.listener;

import com.atguigu.rabbitmq.springbootrabbitmq.config.PublishConfirmConfig;
import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import lombok.extern.slf4j.Slf4j;
import org.springframework.amqp.core.Message;
import org.springframework.amqp.rabbit.annotation.RabbitHandler;
import org.springframework.amqp.rabbit.annotation.RabbitListener;
import org.springframework.stereotype.Component;

import java.net.InetAddress;

/**
 * User: ldj
 * Date: 2023/3/14
 * Time: 8:32
 * Description: No Description
 */
@Slf4j
@Component
@RabbitListener(queues = {PublishConfirmConfig.WARNING_QUEUE_NAME})
public class WarningListener {
    @RabbitHandler
    public void receiveConfirmMsg(Message message, Channel channel, String msg) {
        Connection connection = channel.getConnection();
        InetAddress address = connection.getAddress();
        String hostAddress = address.getHostAddress();

        log.warn("警告! [CF-EXC <{}>] 收到不可路由消息,已转发给备份交换机[BU-EXC]:{}", hostAddress, msg);
    }
}

 3.生产者

package com.atguigu.rabbitmq.springbootrabbitmq.controller;

import com.atguigu.rabbitmq.springbootrabbitmq.config.PublishConfirmConfig;
import com.atguigu.rabbitmq.springbootrabbitmq.config.properties.RabbitmqProperties;
import lombok.extern.slf4j.Slf4j;
import org.springframework.amqp.core.Message;
import org.springframework.amqp.core.MessageProperties;
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.GetMapping;
import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

import java.nio.charset.StandardCharsets;
import java.util.UUID;

/**
 * User: ldj
 * Date: 2023/3/13
 * Time: 18:26
 * Description: No Description
 */
@Slf4j
@RestController
@RequestMapping("/rabbitmq/confirm")
public class ConfirmMessageController {

    @Autowired
    private RabbitTemplate rabbitTemplate;

    // http://localhost:8089/rabbitmq/confirm/sendMsg/发布确认可靠消息

    @GetMapping(value = "/sendMsg/{message}")
    public void sendConfirmMsg(@PathVariable String message) {
        log.info("[sendConfirmMsg]要开始发消息:{}", message);
        MessageProperties messageProperties = new MessageProperties();
        messageProperties.setAppId("666888");

        Message msg = new Message(message.getBytes(StandardCharsets.UTF_8), messageProperties);

        //封装数据给确认回调接口
        CorrelationData correlationData = new CorrelationData();
        correlationData.setId(UUID.randomUUID().toString().replace("-", ""));
        correlationData.setReturnedMessage(msg);
        rabbitTemplate.convertAndSend(PublishConfirmConfig.CONFIRM_EXCHANGE_NAME, RabbitmqProperties.confirmRoutingKey + 1, message, correlationData);
    }
}
package com.atguigu.rabbitmq.springbootrabbitmq.config.properties;

import lombok.Data;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.stereotype.Component;

/**
 * User: ldj
 * Date: 2023/3/12
 * Time: 18:17
 * Description: No Description
 */
@Data
@Component
@ConfigurationProperties("spring.rabbitmq.properties")
public class RabbitmqProperties {

    private String normal1BingingKey;

    private String normal2BingingKey;

    private String normal3BingingKey;

    private String delayedBingingKey;

    private String deadLetterBingingKey;

    private String confirmBingingKey;

    private Integer normal1QueueTtl = 10000;

    private Integer normal2QueueTtl = 30000;


    public static String confirmRoutingKey;

    public void setConfirmRoutingKey(String confirmRoutingKey) {
        RabbitmqProperties.confirmRoutingKey = confirmRoutingKey;
    }
    
}

 测试发现消息是对象时接收到消息不是JSON串,属性是Date类型也没有格式化,所以配上消息转换配置             

package com.atguli.common.config;

import com.fasterxml.jackson.annotation.JsonInclude;
import com.fasterxml.jackson.core.JsonParser;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.module.SimpleModule;
import com.fasterxml.jackson.databind.ser.std.ToStringSerializer;
import org.springframework.amqp.support.converter.Jackson2JsonMessageConverter;
import org.springframework.amqp.support.converter.MessageConverter;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

import java.text.SimpleDateFormat;

/**
 * User: ldj
 * Date: 2022/10/10
 * Time: 17:07
 * Description: rabbitmq序列化配置
 */

@Configuration
public class MyRabbitConfig {

    @Bean
    public MessageConverter messageConverter() {
        ObjectMapper objectMapper = new ObjectMapper();
        objectMapper.setDateFormat(new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"));

        //Long -> String 解决返回前端id是Long类型精度降低,后位数变成0
        SimpleModule simpleModule = new SimpleModule();
        simpleModule.addSerializer(Long.TYPE, ToStringSerializer.instance);
        simpleModule.addSerializer(Long.class, ToStringSerializer.instance);
        objectMapper.registerModule(simpleModule);

        //Include.NON_EMPTY 属性为 空("") 或者为 NULL 都不序列化,忽略该字段
        objectMapper.setSerializationInclusion(JsonInclude.Include.NON_EMPTY);

        // 允许出现单引号
        objectMapper.configure(JsonParser.Feature.ALLOW_SINGLE_QUOTES, true);

        return new Jackson2JsonMessageConverter(objectMapper);
    }
}