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,发布订阅模式比工作队列模式更强大,并且发布订阅模式可以指定自己专用的交换
机
难产难产难产