RabbitMQ---订阅模型-Fanout

1、 订阅模型-Fanout

Fanout,也称为广播。

流程图:

在广播模式下,消息发送流程是这样的:

1) 可以有多个消费者

2) 每个消费者有自己的queue(队列)

3) 每个队列都要绑定到Exchange(交换机)

4) 生产者发送的消息,只能发送到交换机,交换机来决定要发给哪个队列,生产者无法决定。

5) 交换机把消息发送给绑定过的所有队列

6) 队列的消费者都能拿到消息。实现一条消息被多个消费者消费

1.1、生产者

两个变化:

1) 声明Exchange,不再声明Queue

2) 发送消息到Exchange,不再发送到Queue

java 复制代码
public class Send {
   private final static String EXCHANGE_NAME = "fanout_exchange_test";
   public static void main(String[] argv) throws Exception {
       // 获取到连接
       Connection connection = ConnectionUtil.getConnection();
       // 获取通道
       Channel channel = connection.createChannel();
       // 声明exchange,指定类型为fanout
       channel.exchangeDeclare(EXCHANGE_NAME, "fanout");
       // 消息内容
       String message = "Hello everyone";
       // 发布消息到Exchange
       channel.basicPublish(EXCHANGE_NAME, "", null, message.getBytes());
       System.out.println(" [生产者] Sent '" + message + "'");
       channel.close();
       connection.close();
   }
}

1.2、消费者1

java 复制代码
public class Recv {
   private final static String QUEUE_NAME = "fanout_exchange_queue_1";
   private final static String EXCHANGE_NAME = "fanout_exchange_test";
   public static void main(String[] argv) throws Exception {
       // 获取到连接
       Connection connection = ConnectionUtil.getConnection();
       // 获取通道
       Channel channel = connection.createChannel();
       // 声明队列
       channel.queueDeclare(QUEUE_NAME, false, false, false, null);
       // 绑定队列到交换机
       channel.queueBind(QUEUE_NAME, EXCHANGE_NAME, "");
       // 定义队列的消费者
       DefaultConsumer consumer = new DefaultConsumer(channel) {
           // 获取消息,并且处理,这个方法类似事件监听,如果有消息的时候,会被自动调用
           @Override
           public void handleDelivery(String consumerTag, Envelope envelope, BasicProperties properties, byte[] body) throws IOException {
               // body 即消息体
               String msg = new String(body);
               System.out.println(" [消费者1] received : " + msg + "!");
           }
       };
       // 监听队列,自动返回完成
       channel.basicConsume(QUEUE_NAME, true, consumer);
   }
}

1.3、 消费者2

java 复制代码
public class Recv2 {
   private final static String QUEUE_NAME = "fanout_exchange_queue_2";
   private final static String EXCHANGE_NAME = "fanout_exchange_test";
   public static void main(String[] argv) throws Exception {
       // 获取到连接
       Connection connection = ConnectionUtil.getConnection();
       // 获取通道
       Channel channel = connection.createChannel();
       // 声明队列
       channel.queueDeclare(QUEUE_NAME, false, false, false, null);
       // 绑定队列到交换机
       channel.queueBind(QUEUE_NAME, EXCHANGE_NAME, "");
       // 定义队列的消费者
       DefaultConsumer consumer = new DefaultConsumer(channel) {
           // 获取消息,并且处理,这个方法类似事件监听,如果有消息的时候,会被自动调用
           @Override
           public void handleDelivery(String consumerTag, Envelope envelope, BasicProperties properties,
                   byte[] body) throws IOException {
               // body 即消息体
               String msg = new String(body);
               System.out.println(" [消费者2] received : " + msg + "!");
           }
       };
       // 监听队列,手动返回完成
       channel.basicConsume(QUEUE_NAME, true, consumer);
   }
}

1.4、 测试

我们应该先启动生产者,否则,先启动消费者时,由于要绑定交换机,此时,交换机并不存在所以会报错。

我们运行两个消费者,然后发送1条消息:


相关推荐
飞梦工作室几秒前
企业级 Spring Boot 邮件系统开发指南:从基础到高可用架构设计
java·spring boot·后端
haiyu柠檬3 分钟前
在Spring Boot中实现Azure的SSO+VUE3前端配置
java·spring boot·后端
LUCIFER9 分钟前
驱动开发:详细分析 DTB、DTS、DTSI、DTBO 的区别、用途及它们之间的关系
linux·服务器·驱动开发
q***721912 分钟前
springBoot 和springCloud 版本对应关系
spring boot·后端·spring cloud
百***812740 分钟前
【SpringBoot】SpringBoot中分页插件(PageHelper)的使用
java·spring boot·后端
百***864640 分钟前
SpringBoot中自定义Starter
java·spring boot·后端
q***071441 分钟前
VScode 开发 Springboot 程序
java·spring boot·后端
q***465241 分钟前
Spring中使用Async进行异步功能开发实战-以大文件上传为例
java·后端·spring
BullSmall1 小时前
Test Matrix:测试矩阵(IT 领域定义 + 设计实践 + 华为场景应用)
运维·服务器
2501_941147422 小时前
基于 Kotlin 与 Ktor 构建高并发微服务与异步分布式系统实践分享
rabbitmq