RabbitMQ02-RebbitMQ简介及交换器

一. AMQP协议

什么是AMQP协议

AMQP(Advanced Message Queuing Protocol,高级消息队列协议):它是进程之间传递异步消息的网络协议

AMQP工作过程

发布者通过发布消息,通过交换机,交换机根据路由规则将收到的消息分发交换机绑定的下消息队列,最后AMQP代理将消息推送给订阅了此队列的消费者

或消费者按照需求自行获取。

二. RabbitMQ简介

RabbitMQ是通过Erlang语言基于AMQP协议编写的消息中间件,它在分布式系统中可以解应用耦合、流量削峰、异步消息等问题。它有两个特性
队列排队和异步

  1. 应用解耦:多个个应用程序之间可通过RabbitMQ作为媒介,两个应用不再粘连,实现解耦;
  2. 异步消息:多个应用可通过RabbitMQ进行消息传递;
  3. 流量削峰:在高并发情况下,可以通过RabbitMQ的队列特性实现流量削峰;
  4. 应用场景:
    1. 应用到队列特性的应用场景: 排序算法、秒杀活动。
    2. 应用到异步特性的应用场景: 消息分发、异步处理、数据同步、处理耗时任务。

三.springBoot整合RabbitMQ

生产者端发送消息

pom文件

xml 复制代码
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-amqp</artifactId>
            <version>2.6.3</version>
        </dependency>

yml文件

yaml 复制代码
spring:
  application:
    name: producer
  rabbitmq:
    host: xxx
    username: admin
    password: admin

配置类,需要返回一个Queue,org.springframework.amqp.core.Queue下的Queue对象

java 复制代码
@Configuration
public class RabbitMqConfig {

    @Bean
    protected Queue queue(){
        return new Queue("myQueue");
    }
}

使用RabbitMQ发送消息,注入AmqpTemplate,调用convertAndSend()方法

java 复制代码
class ProducerApplicationTests {

    @Autowired
    private AmqpTemplate amqpTemplate;

    @Test
    void send() {
        for (int i = 0; i < 10; i++) {
            amqpTemplate.convertAndSend("myQueue","这是发送的消息");
            System.out.println("发送成功!");
        }

    }

}

消费端接收消息

配置同生产端,不需要配置RabbitMqConfig,接收消息时只需要使用注解RabbitMqConfig,queues属性绑定相应的队列即可。

java 复制代码
@Component
public class ReceiveService {

    @RabbitListener(queues = "myQueue")
    public void test01(String msg){
        System.out.println("接收到消息1" + msg);
    }

    @RabbitListener(queues = "myQueue")
    public void test02(String msg){
        System.out.println("接收到消息2" + msg);
    }

    @RabbitListener(queues = "myQueue")
    public void test03(String msg){
        System.out.println("接收到消息3" + msg);
    }
}

四.交换器(四种)

Direct Exchange:直连交换器

它是RabbitMQ的默认交换器,给指定队列发消息,绑定该消息队列的消费者一次获取消息

实战:

java 复制代码
/** 生产者发送消息,发送10个消息*/
@SpringBootTest
class ProducerApplicationTests {

    @Autowired
    private AmqpTemplate amqpTemplate;

    @Test
    void send() {
        for (int i = 0; i < 10; i++) {
            amqpTemplate.convertAndSend("myQueue","这是发送的消息");
            System.out.println("发送成功!");
        }

    }

}
java 复制代码
/** 接收消息*/
@Component
public class ReceiveService {

    @RabbitListener(queues = "myQueue")
    public void test01(String msg){
        System.out.println("接收到消息1" + msg);
    }

    @RabbitListener(queues = "myQueue")
    public void test02(String msg){
        System.out.println("接收到消息2" + msg);
    }

    @RabbitListener(queues = "myQueue")
    public void test03(String msg){
        System.out.println("接收到消息3" + msg);
    }
}

结果:可以看到1、2、3依次接收消息

shell 复制代码
接收到消息1这是发送的消息
接收到消息2这是发送的消息
接收到消息3这是发送的消息
接收到消息2这是发送的消息
接收到消息3这是发送的消息
接收到消息1这是发送的消息
接收到消息3这是发送的消息
接收到消息1这是发送的消息
接收到消息2这是发送的消息
接收到消息1这是发送的消息

Fanout Exchange:扇形交换器

绑定该交换器的所有队列都可以接收到消息,扇形交换机将消息广播到所有与之绑定的队列。无论消息的路由键是什么,扇形交换机都会将消息发送到所有绑定的队列中。这种类型的交换机常用于实现发布-订阅模式,将消息广播给多个消费者。

实战

java 复制代码
/** 绑定*/
/** Fanout Exchange*/
@Bean
public Queue FanoutExchangeQueue1(){
        return new Queue("fanoutExchangeQueue1");
        }
@Bean
public Queue FanoutExchangeQueue2(){
        return new Queue("fanoutExchangeQueue2");
        }
@Bean
public FanoutExchange fanoutExchange(){
        return new FanoutExchange("amq.fanout");
        }
@Bean
public Binding  FanoutExchangeBinding1(Queue FanoutExchangeQueue1,FanoutExchange fanoutExchange){
        return BindingBuilder.bind(FanoutExchangeQueue1).to(fanoutExchange);
        }
@Bean
public Binding  FanoutExchangeBinding2(Queue FanoutExchangeQueue2,FanoutExchange fanoutExchange){
        return BindingBuilder.bind(FanoutExchangeQueue2).to(fanoutExchange);
        }
java 复制代码
/** 生产者发送消息*/
    @Test
    void sendByFanoutExchange() {
        amqpTemplate.convertAndSend("amq.fanout","key","这是发送到的消息");
        System.out.println("发送成功!");
    }
java 复制代码
    /** 消费者 Direct Exchange*/
    @RabbitListener(queues = "fanoutExchangeQueue1")
    public void test04(String msg){
        System.out.println("接收到消息4" + msg);
    }
    @RabbitListener(queues = "fanoutExchangeQueue2")
    public void test05(String msg){
        System.out.println("接收到消息5" + msg);
    }

结果:每一个绑定到Fanout Exchange上的队列都可以接收到消息

shell 复制代码
接收到消息4这是发送到的消息
接收到消息5这是发送到的消息

Topic Exchange:主题交换器

允许在路由键中设置匹配规则:'*'代表一个字母两个'.'之间的内容;'#'代表0或多个字符;

实战

java 复制代码
    /** 绑定*/
    @Bean
    public Queue topicExchangeQueue1(){
            return new Queue("topicExchangeQueue1");
            }
    @Bean
    public Queue topicExchangeQueue2(){
            return new Queue("topicExchangeQueue2");
            }
    @Bean
    public TopicExchange topicExchange(){
            return new TopicExchange("amq.topic");
            }
    @Bean
    public Binding TopicExchangeToQueue1(Queue topicExchangeQueue1,TopicExchange topicExchange){
            return BindingBuilder.bind(topicExchangeQueue1).to(topicExchange).with("com.shaoby.*");
            }
    @Bean
    public Binding TopicExchangeToQueue2(Queue topicExchangeQueue2,TopicExchange topicExchange){
            return BindingBuilder.bind(topicExchangeQueue2).to(topicExchange).with("com.shaoby.test.#");
    
            }
java 复制代码
    /**生产者发送消息*/
    /** key为com.shaoby.test*/
    @Test
    void sendByTopicExchange() {
            amqpTemplate.convertAndSend("amq.topic","com.shaoby.test","这是发送到的消息");
            System.out.println("发送成功!");
    }
    /** key为com.shaoby.test.a*/
    @Test
    void sendByTopicExchange() {
            amqpTemplate.convertAndSend("amq.topic","com.shaoby.test.a.b","这是发送到的消息");
            System.out.println("发送成功!");
            }
java 复制代码
    /**消费者接收消息*/
    /**Topic Exchange*/
    @RabbitListener(queues = "topicExchangeQueue1")
    public void test06(String msg){
        System.out.println("接收到消息6" + msg);
    }

    @RabbitListener(queues = "topicExchangeQueue2")
    public void test07(String msg){
        System.out.println("接收到消息7" + msg);
    }

结果:

路由key为com.shaoby.test都能接收到消息,com.shaoby.test.a.b只有topicExchangeQueue2能接收到消息

Header Exchange:首部交换器

绑定:

java 复制代码
/** Header Exchange*/
@Bean
public Queue headerExchangeQueue1(){
        return new Queue("headerExchangeQueue1");
        }

@Bean
public Queue headerExchangeQueue2(){
        return new Queue("headerExchangeQueue2");
        }
@Bean
public HeadersExchange headersExchange(){
        return new HeadersExchange("amp.header");
        }
@Bean
public Binding headExchangeToQueue1(Queue headerExchangeQueue1,HeadersExchange headersExchange){
        HashMap<String, Object> map = new HashMap<>();
        map.put("type","OK");
        map.put("status","200");
        return BindingBuilder.bind(headerExchangeQueue1).to(headersExchange).whereAll(map).match();
        }
@Bean
public Binding headExchangeToQueue2(Queue headerExchangeQueue2,HeadersExchange headersExchange){
        HashMap<String, Object> map = new HashMap<>();
        map.put("type","error");
        map.put("status","500");
        return BindingBuilder.bind(headerExchangeQueue2).to(headersExchange).whereAll(map).match();
        }
java 复制代码
/** 生产者发送消息*/
@Test
    void sendByHeadExchange() {
            Map<String, Object> headers = new HashMap<>();
        headers.put("type","OK");
        headers.put("status","200");
        String message = "这是发送到的消息";
        MessageProperties messageProperties = new MessageProperties();
        headers.forEach(messageProperties::setHeader);
        Message msg = new Message(message.getBytes(), messageProperties);
        amqpTemplate.convertAndSend("amp.header",null, msg);
        System.out.println("发送成功!");

        }
java 复制代码
    @RabbitListener(queues = "headerExchangeQueue1")
    public void test08(Message msg){
        System.out.println("接收到消息8:" + msg.toString());

    }
    @RabbitListener(queues = "headerExchangeQueue2")
    public void test09(Message msg){
        System.out.println("接收到消息9:" + msg.toString());
    }

结果:只有匹配上header才能收到消息

shell 复制代码
接收到消息8:(Body:'[B@a7b38a8(byte[24])' MessageProperties [headers={type=OK, status=200}, contentType=application/octet-stream, contentLength=0, receivedDeliveryMode=PERSISTENT, priority=0, redelivered=false, receivedExchange=amp.header, receivedRoutingKey=, deliveryTag=2, consumerTag=amq.ctag-1WTdKW4n_rAEdJUosQD7bg, consumerQueue=headerExchangeQueue1])
相关推荐
.生产的驴26 分钟前
SpringBoot 消息队列RabbitMQ 交换机模式 Fanout广播 Direct定向 Topic话题
spring boot·rabbitmq·java-rabbitmq
王彬泽31 分钟前
【RabbitMQ】消息分发、事务
rabbitmq·事务·消息分发
杰信步迈入C++之路13 小时前
【RabbitMQ】RabbitMQ 概述
分布式·rabbitmq
BLUcoding14 小时前
RabbitMQ08_保证消息可靠性
java·rabbitmq
月夜星辉雪17 小时前
【RabbitMQ 项目】服务端:数据管理模块之消息管理
分布式·rabbitmq
不能再留遗憾了1 天前
RabbitMQ 高级特性——发送方确认
分布式·rabbitmq·ruby
益达_z1 天前
中间件知识点-消息中间件(Rabbitmq)一
分布式·中间件·rabbitmq
.生产的驴1 天前
SpringBoot 消息队列RabbitMQ 消息确认机制确保消息发送成功和失败 生产者确认
java·javascript·spring boot·后端·rabbitmq·负载均衡·java-rabbitmq
.生产的驴1 天前
SpringBoot 消息队列RabbitMQ在代码中声明 交换机 与 队列使用注解创建
java·spring boot·分布式·servlet·kafka·rabbitmq·java-rabbitmq
spiker_1 天前
RabbitMQ 常见使用模式详解
分布式·rabbitmq