RabbitMQ发布订阅模式原理和实现(交换机)

逃离我推掉我的手 2022-01-28 11:25 406阅读 0赞

RabbitMQ发布订阅模式原理和实现(交换机)

这个可能是消息队列中最重要的队列了,其他的都是在它的基础上进行了扩展。
功能实现:一个生产者发送消息,多个消费者获取消息(同样的消息),包括一个生产者,一个交换机,多个队列,多个消费者。
思路解读(重点理解):
(1)一个生产者,多个消费者
(2)每一个消费者都有自己的一个队列
(3)生产者没有直接发消息到队列中,而是发送到交换机
(4)每个消费者的队列都绑定到交换机上
(5)消息通过交换机到达每个消费者的队列
该模式就是Fanout Exchange(扇型交换机)将消息路由给绑定到它身上的所有队列
以用户发邮件案例讲解
注意:交换机没有存储消息功能,如果消息发送到没有绑定消费队列的交换机,消息则丢失。

1179709-20181110225834666-1774156298.png

交换机的作用:

生产者发送消息不会向传统方式直接将消息投递到队列中,而是先将消息投递到交换机中,在由交换机转发到具体的队列,队列在将消息以推送或者拉取方式给消费者进行消费,这和我们之前学习Nginx有点类似。
交换机的作用根据具体的路由策略分发到不同的队列中,交换机有四种类型。
Direct exchange(直连交换机)是根据消息携带的路由键(routing key)将消息投递给对应队列的
Fanout exchange(扇型交换机)将消息路由给绑定到它身上的所有队列
Topic exchange(主题交换机)队列通过路由键绑定到交换机上,然后,交换机根据消息里的路由值,将消息路由给一个或多个绑定队列
Headers exchange(头交换机)类似主题交换机,但是头交换机使用多个消息属性来代替路由键建立路由规则。通过判断消息头的值能否与指定的绑定相匹配来确立路由规则。

Rabbit高级队列(发布订阅)

生产者投递消息给交换机缓存起来(不会直接给队列),交换机根据路由策略RoutingKey转发到不同的队列服务器中。队列服务器再以推送或者拉取形式让消费者消费。(类似Nginx)

RabbitMQ发布与订阅原理:

案例: 用户注册 —-> 发送邮件 —->发送短信

1179709-20181110234122192-499747803.png


pom文件:

  1. <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
  2. <modelVersion>4.0.0</modelVersion>
  3. <groupId>com.toov5.rabibitMQScribe</groupId>
  4. <artifactId>rabibitMQScribe</artifactId>
  5. <version>0.0.1-SNAPSHOT</version>
  6. <dependencies>
  7. <dependency>
  8. <groupId>com.rabbitmq</groupId>
  9. <artifactId>amqp-client</artifactId>
  10. <version>3.6.5</version>
  11. </dependency>
  12. </dependencies>
  13. </project>

 连接工具类:

  1. package com.toov5.utils;
  2. import java.io.IOException;
  3. import java.util.concurrent.TimeoutException;
  4. import com.rabbitmq.client.Connection;
  5. import com.rabbitmq.client.ConnectionFactory;
  6. //没有做成单例的 VirtualHost 需要复用
  7. public class MQConnectionUtils {
  8. //创建新的连接
  9. public static Connection newConnection() throws IOException, TimeoutException {
  10. //创建连接工厂
  11. ConnectionFactory factory= new ConnectionFactory();
  12. //链接地址
  13. factory.setHost("192.168.91.6");
  14. //用户名称
  15. factory.setUsername("admin");
  16. //用户密码
  17. factory.setPassword("admin");
  18. //amqp端口号
  19. factory.setPort(5672);
  20. //连接virtualhost
  21. factory.setVirtualHost("/admin_toov5");
  22. Connection connection = factory.newConnection();
  23. return connection;
  24. }
  25. }

生产者:

  1. package com.toov5.fanout;
  2. import java.io.IOException;
  3. import java.util.concurrent.TimeoutException;
  4. import com.rabbitmq.client.Channel;
  5. import com.rabbitmq.client.Connection;
  6. import com.toov5.utils.MQConnectionUtils;
  7. //生产者 交换机类型 producerFanout类型
  8. public class FanoutProducer {
  9. //交换机名称
  10. private static final String EXCHANGE_NAME = "my_fanout";
  11. public static void main(String[] args) throws IOException, TimeoutException {
  12. //建立MQ连接
  13. Connection connection = MQConnectionUtils.newConnection();
  14. //创建通道
  15. Channel channel = connection.createChannel();
  16. //生产者绑定交换机
  17. channel.exchangeDeclare(EXCHANGE_NAME, "fanout"); //交换机名称 交换机类型
  18. //创建对应的消息
  19. String msString = "my_fanout_destination_msg";
  20. //通过频道 发送消息
  21. System.out.println("生产者投递消息:"+msString);
  22. //消息投递到交换机里面去
  23. channel.basicPublish(EXCHANGE_NAME, "", null, msString.getBytes());
  24. //关闭通道 和 连接
  25. channel.close();
  26. connection.close();
  27. }
  28. }

消费者:

  1. package com.toov5.fanout;
  2. import java.io.IOException;
  3. import java.util.concurrent.TimeoutException;
  4. import com.rabbitmq.client.Channel;
  5. import com.rabbitmq.client.Connection;
  6. import com.rabbitmq.client.DefaultConsumer;
  7. import com.rabbitmq.client.Envelope;
  8. import com.rabbitmq.client.AMQP.BasicProperties;
  9. import com.toov5.utils.MQConnectionUtils;
  10. //邮件消费者
  11. public class ConsumerEmailFanout {
  12. private static final String EMAIL_QUEUE ="email_queue_fanout";
  13. //交换机名称
  14. private static final String EXCHANGE_NAME = "my_fanout";
  15. public static void main(String[] args) throws IOException, TimeoutException {
  16. System.out.println("邮件消费者启动");
  17. //建立MQ连接
  18. Connection connection = MQConnectionUtils.newConnection();
  19. //创建通道
  20. Channel channel = connection.createChannel();
  21. //消费者声明队列
  22. channel.queueDeclare(EMAIL_QUEUE, false, false, false, null);
  23. //消费者队列绑定交换机
  24. channel.queueBind(EMAIL_QUEUE, EXCHANGE_NAME, "");
  25. //消费者监听消息
  26. DefaultConsumer defaultConsumer = new DefaultConsumer(channel) {
  27. //重写监听方法
  28. @Override
  29. public void handleDelivery(String consumerTag, Envelope envelope, BasicProperties properties, byte[] body)
  30. throws IOException {
  31. String msg = new String(body,"UTF-8");
  32. System.out.println("邮件消费者获取生产者消息"+msg);
  33. }
  34. };
  35. channel.basicConsume(EMAIL_QUEUE,true, defaultConsumer); //绑定队列 事件监听
  36. }
  37. }
  38. package com.toov5.fanout;
  39. import java.io.IOException;
  40. import java.util.concurrent.TimeoutException;
  41. import com.rabbitmq.client.Channel;
  42. import com.rabbitmq.client.Connection;
  43. import com.rabbitmq.client.DefaultConsumer;
  44. import com.rabbitmq.client.Envelope;
  45. import com.rabbitmq.client.AMQP.BasicProperties;
  46. import com.toov5.utils.MQConnectionUtils;
  47. //邮件消费者
  48. public class ConsumerSMSFanout {
  49. private static final String SMS_QUEUE ="sms_queue_fanout";
  50. //交换机名称
  51. private static final String EXCHANGE_NAME = "my_fanout";
  52. public static void main(String[] args) throws IOException, TimeoutException {
  53. System.out.println("短信消费者启动");
  54. //建立MQ连接
  55. Connection connection = MQConnectionUtils.newConnection();
  56. //创建通道
  57. Channel channel = connection.createChannel();
  58. //消费者声明队列
  59. channel.queueDeclare(SMS_QUEUE, false, false, false, null);
  60. //消费者队列绑定交换机
  61. channel.queueBind(SMS_QUEUE, EXCHANGE_NAME, "");
  62. //消费者监听消息
  63. DefaultConsumer defaultConsumer = new DefaultConsumer(channel) {
  64. //重写监听方法
  65. @Override
  66. public void handleDelivery(String consumerTag, Envelope envelope, BasicProperties properties, byte[] body)
  67. throws IOException {
  68. String msg = new String(body,"UTF-8");
  69. System.out.println("邮件消费者获取生产者消息"+msg);
  70. }
  71. };
  72. channel.basicConsume(SMS_QUEUE,true, defaultConsumer); //绑定队列 事件监听
  73. }
  74. }

1179709-20181111203019716-1546664035.png

1179709-20181111204144146-1064416481.png

1179709-20181111204203066-696236354.png

1179709-20181111204215007-1216368916.png

发表评论

表情:
评论列表 (有 0 条评论,406人围观)

还没有评论,来说两句吧...

相关阅读

    相关 RabbitMQ:发布订阅模式

    P:生产者,发送消息给交换机C:消费者,接收消息X:交换机,一方面接收生产者发送的消息,另一方面知道怎么处理消息,是否应将其附加到特定队列?是否应将其附加到多个队列中?或...