概述

Fanout Exchange 会将接收到的消息广播到每一个跟其绑定的queue

image-1655735310859

生产者将消息发送到交换机,由交换机再去发给绑定该交换机的所有队列,每个队列接受到的消息都是一样的,再有消费者去消费,注意,此处的交换机exchange并不存储消息,存储消息的是队列queue。

SpringAMQP提供了声明交换机、队列、绑定关系的API,例如:

image-1655735767692

SpringAMQP实现

首先需要定义个配置类,来维护队列和交换机的关系
在consumer服务常见一个类,添加@Configuration注解,并声明FanoutExchange、Queue和绑定关系对象Binding,代码如下

@Configuration
public class FanoutConfig {

    //交换机
    @Bean
    public FanoutExchange fanoutExchange() {
        return new FanoutExchange("fanout.exchange");
    }

    //队列1
    @Bean
    public Queue fanoutQueue1() {
        return new Queue("fanout.queue1");
    }

    //绑定队列1到交换机
    @Bean
    public Binding bindingQueue1(Queue fanoutQueue1, FanoutExchange fanoutExchange) {
        return BindingBuilder
                .bind(fanoutQueue1)
                .to(fanoutExchange);
    }

    //队列2
    @Bean
    public Queue fanoutQueue2() {
        return new Queue("fanout.queue2");
    }

    //绑定队列2到交换机
    @Bean
    public Binding bindingQueue2(Queue fanoutQueue2, FanoutExchange fanoutExchange) {
        return BindingBuilder
                .bind(fanoutQueue2)
                .to(fanoutExchange);
    }

}

消费者
在consumer服务的SpringRabbitListener类中,添加两个方法,分别监听fanout.queue1和fanout.queue2:

@Component
public class SpringRabbitListener {

    @RabbitListener(queues = "fanout.queue1")
    public void listenFanoutQueue1(String msg) {
        System.out.println("【fanout.queue1】:" + msg);
    }

    @RabbitListener(queues = "fanout.queue2")
    public void listenFanoutQueue2(String msg) {
        System.err.println("【fanout.queue2】:" + msg);
    }
}

生产者

@RunWith(SpringRunner.class)
@SpringBootTest
public class SpringAmqpTest {
    @Autowired
    private RabbitTemplate template;

    @Test
    public void testSendFanoutExchange() {
        String exchangeName = "fanout.exchange";
        String message = "hello every one !";
        template.convertAndSend(exchangeName, "", message);
    }
}

测试

将消费者启动起来,再讲生产者的消息生产方法启动,发现两个队列都可以收到消息:

image-1655735690092

总结

交换机的作用是什么?

  • 接收publisher发送的消息
  • 将消息按照规则路由到与之绑定的队列
  • 不能缓存消息,路由失败,消息丢失
  • FanoutExchange的会将消息路由到每个绑定的队列

声明队列、交换机、绑定关系的Bean是什么?

  • Queue
  • FanoutExchange
  • Binding