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

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

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

交换机的作用:

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

 

 

Rabbit高级队列(发布订阅)

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

  

RabbitMQ发布与订阅原理: 

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

      

  


 

 pom文件:

<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">
  <modelVersion>4.0.0</modelVersion>
  <groupId>com.toov5.rabibitMQScribe</groupId>
  <artifactId>rabibitMQScribe</artifactId>
  <version>0.0.1-SNAPSHOT</version>
  
  <dependencies>
		<dependency>
			<groupId>com.rabbitmq</groupId>
			<artifactId>amqp-client</artifactId>
			<version>3.6.5</version>
		</dependency>
	</dependencies>
  
</project>

 连接工具类:

package com.toov5.utils;

import java.io.IOException;
import java.util.concurrent.TimeoutException;

import com.rabbitmq.client.Connection;
import com.rabbitmq.client.ConnectionFactory;
 
//没有做成单例的  VirtualHost 需要复用
public class MQConnectionUtils {
    //创建新的连接
    public static Connection newConnection() throws IOException, TimeoutException {
         //创建连接工厂
    ConnectionFactory factory= new ConnectionFactory();
    //链接地址
    factory.setHost("192.168.91.6");
    //用户名称
    factory.setUsername("admin");
    //用户密码
    factory.setPassword("admin");
    //amqp端口号
    factory.setPort(5672);
    //连接virtualhost
    factory.setVirtualHost("/admin_toov5");
    Connection connection = factory.newConnection();
        return connection;
    }    
}

 

生产者:

package com.toov5.fanout;

import java.io.IOException;
import java.util.concurrent.TimeoutException;

import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import com.toov5.utils.MQConnectionUtils;

//生产者 交换机类型 producerFanout类型
public class FanoutProducer {
    //交换机名称
     private static final String EXCHANGE_NAME = "my_fanout"; 
     public static void main(String[] args) throws IOException, TimeoutException {
        //建立MQ连接
         Connection connection = MQConnectionUtils.newConnection();
        //创建通道
          Channel channel = connection.createChannel();
          //生产者绑定交换机
          channel.exchangeDeclare(EXCHANGE_NAME, "fanout");  //交换机名称  交换机类型
          //创建对应的消息 
          String msString = "my_fanout_destination_msg";
          //通过频道 发送消息
          System.out.println("生产者投递消息:"+msString);
          //消息投递到交换机里面去
          channel.basicPublish(EXCHANGE_NAME, "", null, msString.getBytes());
          //关闭通道 和 连接
          channel.close();
          connection.close();
    }
    
} 

消费者:

package com.toov5.fanout;

import java.io.IOException;
import java.util.concurrent.TimeoutException;

import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import com.rabbitmq.client.DefaultConsumer;
import com.rabbitmq.client.Envelope;
import com.rabbitmq.client.AMQP.BasicProperties;
import com.toov5.utils.MQConnectionUtils;

//邮件消费者
public class ConsumerEmailFanout {
    private static final String EMAIL_QUEUE ="email_queue_fanout";
    //交换机名称
   private static final String EXCHANGE_NAME = "my_fanout"; 
     public static void main(String[] args) throws IOException, TimeoutException {
         System.out.println("邮件消费者启动");
        //建立MQ连接
         Connection connection = MQConnectionUtils.newConnection();
        //创建通道
          Channel channel = connection.createChannel();
          
       //消费者声明队列
          channel.queueDeclare(EMAIL_QUEUE, false, false, false, null);
         //消费者队列绑定交换机
          channel.queueBind(EMAIL_QUEUE, EXCHANGE_NAME, "");
          //消费者监听消息
      DefaultConsumer defaultConsumer =     new DefaultConsumer(channel) {
              //重写监听方法
             @Override
             public void handleDelivery(String consumerTag, Envelope envelope, BasicProperties properties, byte[] body)
                     throws IOException {             
                 String msg = new String(body,"UTF-8");
                 System.out.println("邮件消费者获取生产者消息"+msg);
             }
         };
         channel.basicConsume(EMAIL_QUEUE,true, defaultConsumer);   //绑定队列 事件监听
            
    }
}
package com.toov5.fanout;

import java.io.IOException;
import java.util.concurrent.TimeoutException;

import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import com.rabbitmq.client.DefaultConsumer;
import com.rabbitmq.client.Envelope;
import com.rabbitmq.client.AMQP.BasicProperties;
import com.toov5.utils.MQConnectionUtils;

//邮件消费者
public class ConsumerSMSFanout {
    private static final String SMS_QUEUE ="sms_queue_fanout";
    //交换机名称
   private static final String EXCHANGE_NAME = "my_fanout"; 
     public static void main(String[] args) throws IOException, TimeoutException {
         System.out.println("短信消费者启动");
        //建立MQ连接
         Connection connection = MQConnectionUtils.newConnection();
        //创建通道
          Channel channel = connection.createChannel();
          
       //消费者声明队列
          channel.queueDeclare(SMS_QUEUE, false, false, false, null);
         //消费者队列绑定交换机
          channel.queueBind(SMS_QUEUE, EXCHANGE_NAME, "");
          //消费者监听消息
      DefaultConsumer defaultConsumer = new DefaultConsumer(channel) {
              //重写监听方法
             @Override
             public void handleDelivery(String consumerTag, Envelope envelope, BasicProperties properties, byte[] body)
                     throws IOException {             
                 String msg = new String(body,"UTF-8");
                 System.out.println("邮件消费者获取生产者消息"+msg);
             }
         };
         channel.basicConsume(SMS_QUEUE,true, defaultConsumer);   //绑定队列 事件监听
  
    }
}

 

 

 

  • 0
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
RabbitMQ是一个使用AMQP(Advanced Message Queuing Protocol,高级消息队列协议)实现的开源消息队列系统。它基于消息队列的概念,提供了一种可靠的、可扩展的、灵活的消息传递模式。 底层原理: 1. Producer(生产者)使用AMQP协议将消息发送到RabbitMQ的Exchange(交换机)。 2. Exchange根据预设的规则(如路由键)将消息路由到一个或多个Queue(队列)。 3. Consumer(消费者)订阅并从队列中接收消息进行处理。 RabbitMQ实现方法主要包括以下几个关键组件: 1. Exchange(交换机):接收Producer发送的消息,并根据预设的规则将消息路由到一个或多个Queue。常见的Exchange类型有Direct、Topic、Fanout和Headers。 2. Queue(队列):是消息的容器,可以持久化存储消息,Consumer从队列中获取消息进行消费。每个队列都有一个名称,并且可以绑定多个Exchange。 3. Binding(绑定):用于将Exchange和Queue进行绑定,绑定时可以指定特定的规则,例如路由键、Headers等。 4. Routing Key(路由键):Producer发送消息时,可以指定一个路由键,Exchange根据路由键将消息发送到匹配的Queue。 5. AMQP协议:RabbitMQ使用AMQP协议作为其通信协议,该协议定义了Producer、Exchange、Queue和Consumer之间的通信方式和规则。 通过以上组件的配合,RabbitMQ实现了高效、可靠的消息传递机制。它支持多种消息传递模式,例如点对点(Point-to-Point)和发布/订阅(Publish/Subscribe),并且具有可靠性、灵活性和可扩展性等优势。 需要注意的是,RabbitMQ还提供了各种高级特性,如消息的持久化、消息的优先级、消息确认机制等,以满足不同场景下的需求。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值