【转】rabbitmq-fanout(广播模式)
版权声明:本文为CSDN博主「fakerswe」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/fakerswe/article/details/81455340
————————————————
一、什么是fanout(广播模式)
简单的讲,就是把交换机(Exchange)里的消息发送给所有绑定该交换机的队列,忽略routingKey。

由图可知,生产者把消息发送到交换机后,由交换机发送给消费者队列。消费者队列如果想要接收到交换机里的消息,那么需要保证:队列绑定的交换机名称要和交换机一致,这个是广播模式的关键,也是MQ后续所有模式最粗略的前提。
比如:
1)生产者声明一个交换机(Exchange),交换机名称为“fanoutLogs”,类型为广播模式“fanout”,消息为“Now you see me”;
channel.exchangeDeclare("fanoutLogs", "fanout");
2)消费者声明一个队列Q1,声明一个交换机(名称为“fanoutLogs”,类型为广播模式”fanout“),最后队列和交换机绑定;
channel.exchangeDeclare("fanoutLogs", "fanout"); channel.queueBind("Q1", "fanoutLogs", "");
3)消费者声明另一个队列Q2,声明一个交换机(名称为”phantaciLogs“,类型为广播模式”fanout“),最后队列与交换机绑定;
channel.exchangeDeclare("phantaciLogs", "fanout"); channel.queueBind("Q2", "phantaciLogs", "");
4)需注意,我们要先启动声明消费者,再启动声明生产者。否则先启动生产者的话,exchange接到消息后发现没有队列对它感兴趣,就任性的把消息给丢掉了;
5)Q1和Q2两个消费者启动后,启动生产者。可以发现,和生产者交换机(Exchange)名称相同的Q1正常接收到消息;Q2虽然也是广播模式(fanout),但交换机名称不同,所以未接收到消息。
RabbitMQ消息模型的核心思想(core idea):
生产者会把消息发送给RabbitMQ的交换中心(Exchange),Exchange的一侧是生产者,另一侧则是一个或多个队列,由Exchange决定一条消息的生命周期–发送给某些队列,或者直接丢弃掉。
二、代码域
1. 生产者【FanoutBoss】
package com.iyungu.phantaci.test.rabbitmq; import com.rabbitmq.client.*; import org.apache.log4j.Logger; import java.io.IOException; import java.util.concurrent.TimeoutException; //广播模式-消息生产者 public class FanoutBoss { private static final Logger logger = Logger.getLogger(FanoutBoss.class); public static void main(String[] args) { //New一个RabbitMQ的连接工厂 ConnectionFactory factory = new ConnectionFactory(); //设置需要连接的RabbitMQ地址,这里指向本机 factory.setHost("localhost"); try { //尝试获取一个连接 Connection connection = factory.newConnection(); //尝试创建一个channel Channel channel = connection.createChannel(); String message = "当前时间为:2018年8月6日14:25:14"; //声明交换机(参数为:交换机名称; 交换机类型,广播模式) channel.exchangeDeclare("fanoutLogs", BuiltinExchangeType.FANOUT); //消息发布(参数为:交换机名称; routingKey,忽略。在广播模式中,生产者声明交换机的名称和类型即可) channel.basicPublish("fanoutLogs","", null,message.getBytes()); logger.info("********Message********:发送成功"); channel.close(); connection.close(); } catch (IOException |TimeoutException e) { e.printStackTrace(); } } }
2. 消费者【FanoutWorker】
package com.iyungu.phantaci.test.rabbitmq; import com.rabbitmq.client.*; import org.apache.log4j.Logger; import java.io.IOException; import java.util.concurrent.TimeoutException; //广播模式-消息消费者 public class FanoutWorker { private static final Logger logger = Logger.getLogger(FanoutWorker.class); public static void main(String[] args) { ConnectionFactory factory = new ConnectionFactory(); factory.setHost("localhost"); try { Connection connection = factory.newConnection(); Channel channel = connection.createChannel(); //交换机声明(参数为:交换机名称;交换机类型) channel.exchangeDeclare("fanoutLogs",BuiltinExchangeType.FANOUT); //获取一个临时队列 String queueName = channel.queueDeclare().getQueue(); //队列与交换机绑定(参数为:队列名称;交换机名称;routingKey忽略) channel.queueBind(queueName,"fanoutLogs",""); logger.info("********Waiting for messages********"); //这里重写了DefaultConsumer的handleDelivery方法,因为发送的时候对消息进行了getByte(),在这里要重新组装成String Consumer consumer = new DefaultConsumer(channel) { @Override public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException { super.handleDelivery(consumerTag, envelope, properties, body); String message = new String(body,"UTF-8"); logger.info("received:" + message); } }; //声明队列中被消费掉的消息(参数为:队列名称;消息是否自动确认;consumer主体) channel.basicConsume(queueName,true,consumer); //这里不能关闭连接,调用了消费方法后,消费者会一直连接着rabbitMQ等待消费 } catch (IOException |TimeoutException e) { e.printStackTrace(); } } }

浙公网安备 33010602011771号