RabbitMQ工作模式-工作队列模式
RabbitMQ有以下几种工作模式:
1,Work Queues 工作队列
2,Publish/Subscribe 发布订阅
3,Routing 路由
4,Topics 通配符
5,Header Header转换器
6,RPC 远程过程调用
1.Work Queues 工作队列

work queues与入门程序相比,多了一个消费端,两个消费端共同消费同一个队列中的消息。
应用场景:对于 任务过重或任务较多情况使用工作队列可以提高任务处理的速度。
消息不能被重复消费
测试:
1、使用入门程序,启动多个消费者。
2、生产者发送多个消息。
结果:
1、一条消息只会被一个消费者接收;
2、rabbit采用轮询的方式将消息是平均发送给消费者的;
3、消费者在处理完某条消息后,才会收到下一条消息。
2.Publish/subscribe 发布订阅模式

发布订阅模式:
1、每个消费者监听自己的队列。
2、生产者将消息发给broker,由交换机将消息转发到绑定此交换机的每个队列,每个绑定交换机的队列都将接收到消息
使用的代码案例(用户通知:有短信通知,邮件,电话等等):
生产者:
public class Producer02_publish {
//private static final String QUEUE = "HEllO";
//队列名
private static final String QUEUE_INFORM_EMAIL = "queue_inform_email";
private static final String QUEUE_INFORM_SMS = "queue_inform_sms";
private static final String EXCHANGE_FANOUT_INFORM = "exchange_fanout_inform";
public static void main(String[] args) throws IOException, TimeoutException {
//生产者于MQ建立连接
//通过建立连接工厂建立连接
ConnectionFactory connectionFactory = new ConnectionFactory();
//配置连接信息
connectionFactory.setHost("localhost");
connectionFactory.setPort(5672); //端口
connectionFactory.setUsername("guest");
connectionFactory.setPassword("guest");
//设置虚拟机 实现多个虚拟MQ 每个虚拟机相当于一个独立的MQ
connectionFactory.setVirtualHost("/");
//建立新连接
Connection connection = null;
Channel channel = null;
try {
connection = connectionFactory.newConnection();
//创建会话通道 生产者和mq服务所有通信都在通道中完成
channel = connection.createChannel();
/** 声明队列 如果队列在MQ中没有,则创建
queueDeclare(String queue, boolean durable, boolean exclusive, boolean autoDelete, Map<String, Object> arguments)
param1:队列名称
* param2:是否持久化 (如果重启后,队列还在)
* param3:队列是否独占此连接-------队列只允许在该连接中访问,连接关闭队列删除
* param4:队列不再使用时是否自动删除此队列
* param5:队列参数--------队列拓展参数
* */
channel.queueDeclare(QUEUE_INFORM_EMAIL,true,false,false,null);
channel.queueDeclare(QUEUE_INFORM_SMS,true,false,false,null);
/**
* 声明交换机
* param1:交换机名称
* param2:交换机类型
* fanout:对应的工作模式是:发布订阅publish/scribe
* direct:对应Routing工作模式
* topic:对应通配符工作模式
* headers:对应headers模式
*/
channel.exchangeDeclare(EXCHANGE_FANOUT_INFORM, BuiltinExchangeType.FANOUT);
//交换机和队列绑定String queue, String exchange, String routingKey
/**
* 参数明细
* 1、队列名称
* 2、交换机名称
* 3、路由key 交换机根据路由KEY的值将消息转发到指定队列 在发布订阅工作模式时设置为空字符串
*/
channel.queueBind(QUEUE_INFORM_EMAIL,EXCHANGE_FANOUT_INFORM,"");
channel.queueBind(QUEUE_INFORM_SMS,EXCHANGE_FANOUT_INFORM,"");
for (int i = 0; i < 5; i++) {
//定义一个消息的内容
String message = "Hello,Send inform message to User";
//发送消息
/**
* void basicPublish(String exchange, String routingKey, AMQP.BasicProperties props, byte[] body)
* param1:Exchange的名称,如果没有指定,则使用Default Exchange
* param2:routingKey,消息的路由Key,是用于Exchange(交换机)将消息转发到指定的消息队列,(如果是默认路由,routingKEY使用队列名)
* param3:消息包含的属性
* param4:消息体
* */
channel.basicPublish(EXCHANGE_FANOUT_INFORM,"",null,message.getBytes());
System.out.println("Send to mq :'" + message + "'");
}
} catch (Exception e) {
e.printStackTrace();
} finally {
if(channel != null)
{
channel.close();
}
if(connection != null)
{
connection.close();
}
}
}
}
消费者:
public class Consumer02_subscribe_email {
//队列名
private static final String QUEUE_INFORM_EMAIL = "queue_inform_email";
private static final String EXCHANGE_FANOUT_INFORM = "exchange_fanout_inform";
public static void main(String[] args) throws IOException, TimeoutException {
//消费者于MQ建立连接
//通过建立连接工厂建立连接
ConnectionFactory connectionFactory = new ConnectionFactory();
//配置连接信息
connectionFactory.setHost("localhost");
connectionFactory.setPort(5672); //端口
connectionFactory.setUsername("guest");
connectionFactory.setPassword("guest");
//设置虚拟机 实现多个虚拟MQ 每个虚拟机相当于一个独立的MQ
connectionFactory.setVirtualHost("/");
//建立新连接
Connection connection = connectionFactory.newConnection();
//创建会话通道
Channel channel = connection.createChannel();
//声明队列 如果队列在MQ中没有,则创建
/** queueDeclare(String queue, boolean durable, boolean exclusive, boolean autoDelete, Map<String, Object> arguments)
param1:队列名称
* param2:是否持久化 (如果重启后,队列还在)
* param3:队列是否独占此连接-------队列只允许在该连接中访问,连接关闭队列删除
* param4:队列不再使用时是否自动删除此队列
* param5:队列参数--------队列拓展参数
* */
channel.queueDeclare(QUEUE_INFORM_EMAIL, true, false, false, null);
/**
* 声明交换机
* param1:交换机名称
* param2:交换机类型
* fanout:对应的工作模式是:发布订阅publish/scribe
* direct:对应Routing工作模式
* topic:对应通配符工作模式
* headers:对应headers模式
*/
channel.exchangeDeclare(EXCHANGE_FANOUT_INFORM, BuiltinExchangeType.FANOUT);
//交换机和队列绑定String queue, String exchange, String routingKey
/**
* 参数明细
* 1、队列名称
* 2、交换机名称
* 3、路由key 交换机根据路由KEY的值将消息转发到指定队列 在发布订阅工作模式时设置为空字符串
*/
channel.queueBind(QUEUE_INFORM_EMAIL,EXCHANGE_FANOUT_INFORM,"");
//实现消费方法
DefaultConsumer defaultConsumer = new DefaultConsumer(channel) {
/**
当接收到消息后,此方法被调用
* @param consumerTag 消费者标签:用来标识消费者
* @param envelope 信封:可以拿到信息
* @param properties 消息属性
* @param body 消息内容
* @throws IOException
*/
@Override
public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {
//super.handleDelivery(consumerTag, envelope, properties, body);
//交换机
String exchange = envelope.getExchange();
//消息ID:在channel中标识消息ID,可用于消息被确认接受
long deliveryTag = envelope.getDeliveryTag();
//消息内容
String message = new String(body,"UTF-8");
System.out.println("receive a message: " + message);
}
};
//监听队列
/**
* 监听队列String queue, boolean autoAck,Consumer callback
* 参数明细
* 1、队列名称
* 2、是否自动回复,设置为true为表示消息接收到自动向mq回复接收到了,mq接收到回复会删除消息,设置
为false则需要手动回复
* 3、消费消息的方法,消费者接收到消息后调用此方法
*/
channel.basicConsume(QUEUE_INFORM_EMAIL, true, defaultConsumer);
}
}
消费者2:
public class Consumer02_subscribe_sms {
//队列名
private static final String QUEUE_INFORM_SMS = "queue_inform_sms";
private static final String EXCHANGE_FANOUT_INFORM = "exchange_fanout_inform";
public static void main(String[] args) throws IOException, TimeoutException {
//消费者于MQ建立连接
//通过建立连接工厂建立连接
ConnectionFactory connectionFactory = new ConnectionFactory();
//配置连接信息
connectionFactory.setHost("localhost");
connectionFactory.setPort(5672); //端口
connectionFactory.setUsername("guest");
connectionFactory.setPassword("guest");
//设置虚拟机 实现多个虚拟MQ 每个虚拟机相当于一个独立的MQ
connectionFactory.setVirtualHost("/");
//建立新连接
Connection connection = connectionFactory.newConnection();
//创建会话通道
Channel channel = connection.createChannel();
//声明队列 如果队列在MQ中没有,则创建
/** queueDeclare(String queue, boolean durable, boolean exclusive, boolean autoDelete, Map<String, Object> arguments)
param1:队列名称
* param2:是否持久化 (如果重启后,队列还在)
* param3:队列是否独占此连接-------队列只允许在该连接中访问,连接关闭队列删除
* param4:队列不再使用时是否自动删除此队列
* param5:队列参数--------队列拓展参数
* */
channel.queueDeclare(QUEUE_INFORM_SMS, true, false, false, null);
/**
* 声明交换机
* param1:交换机名称
* param2:交换机类型
* fanout:对应的工作模式是:发布订阅publish/scribe
* direct:对应Routing工作模式
* topic:对应通配符工作模式
* headers:对应headers模式
*/
channel.exchangeDeclare(EXCHANGE_FANOUT_INFORM, BuiltinExchangeType.FANOUT);
//交换机和队列绑定String queue, String exchange, String routingKey
/**
* 参数明细
* 1、队列名称
* 2、交换机名称
* 3、路由key 交换机根据路由KEY的值将消息转发到指定队列 在发布订阅工作模式时设置为空字符串
*/
channel.queueBind(QUEUE_INFORM_SMS,EXCHANGE_FANOUT_INFORM,"");
//实现消费方法
DefaultConsumer defaultConsumer = new DefaultConsumer(channel) {
/**
当接收到消息后,此方法被调用
* @param consumerTag 消费者标签:用来标识消费者
* @param envelope 信封:可以拿到信息
* @param properties 消息属性
* @param body 消息内容
* @throws IOException
*/
@Override
public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {
//super.handleDelivery(consumerTag, envelope, properties, body);
//交换机
String exchange = envelope.getExchange();
//消息ID:在channel中标识消息ID,可用于消息被确认接受
long deliveryTag = envelope.getDeliveryTag();
//消息内容
String message = new String(body,"UTF-8");
System.out.println("receive a message: " + message);
}
};
//监听队列
/**
* 监听队列String queue, boolean autoAck,Consumer callback
* 参数明细
* 1、队列名称
* 2、是否自动回复,设置为true为表示消息接收到自动向mq回复接收到了,mq接收到回复会删除消息,设置
为false则需要手动回复
* 3、消费消息的方法,消费者接收到消息后调用此方法
*/
channel.basicConsume(QUEUE_INFORM_SMS, true, defaultConsumer);
}
}
1、publish/subscribe与work queues有什么区别。
区别:
1)work queues不用定义交换机,而publish/subscribe需要定义交换机。
2)publish/subscribe的生产方是面向交换机发送消息,work queues的生产方是面向队列发送消息(底层使用默认
交换机)。
3)publish/subscribe需要设置队列和交换机的绑定,work queues不需要设置,实质上work queues会将队列绑
定到默认的交换机 。
相同点:
所以两者实现的发布/订阅的效果是一样的,多个消费端监听同一个队列不会重复消费消息。
2、实质工作用什么 publish/subscribe还是work queues。
建议使用 publish/subscribe,发布订阅模式比工作队列模式更强大,并且发布订阅模式可以指定自己专用的交换
机
难产难产难产

浙公网安备 33010602011771号