前言  

   AMQP,即Advanced Message Queuing Protocol,高级消息队列协议,是应用层协议的一个开放标准,为面向消息的中间件设计。消息中间件主要用于组件之间的解耦。

业务需求

   后端(集群)通过websocket往各自维持的websocket session推送消息,如果采用每个实例监听同一个queue,那么生产者往该queue中推送一条消息,该消息只能被集群中某个实例消费一次。

   想要实现后端每个实例同时消费该消息,便可采用RabbitMQ中的topic模式,即每个实例启动时,新建一个topic类型的exchange,routing key为"#.queue",并且每个实例的queue与该exchange绑定。每个实例中,根据hostname+".queue"(queue为配置文件中指定的默认队列名称)来创建各自的queue,这样每个实例都各自监听自己的queue。

   生产者往该exchange中推送消息,routing key使用默认队列(或者每台实例自己的队列都可以,只要能匹配到"#.queue"即可),这样exchange接收到消息后,会根据routingkey来匹配与该exchange绑定的queues,并将消息发送到符合条件的queues中,这样每台实例都能收到该消息并消费。

RabbitMQ介绍

   RabbitMQ是一个开源的AMQP实现,服务器端用Erlang语言编写,用于在分布式系统中存储转发消息,在易用性、扩展性、高可用性等方面表现不俗。

整体结构图

    

概念介绍
  • Broker:简单来说就是消息队列服务器实体
  • Virtual Host:虚拟主机,一个Broker里可以开设多个virtual host,用作不同用户的权限分离
  • Exchange:消息交换机,它指定消息按什么规则,路由到哪个队列
  • Queue:消息队列载体,每个消息都会被投入到一个或多个队列
  • Binding:绑定,它的作用就是把exchange和queue按照路由规则绑定起来
  • Routing Key:路由关键字,exchange根据这个关键字进行消息投递
  • Publisher:消息生产者
  • Consumer:消息消费者
  • Channel:消息通道,在客户端的每个连接里,可建立多个channel,每个channel代表一个会话任务
Exchange详解
   Exchange意思为交换机,从图中可以看出,publisher发送消息,先进入Exchange,然后由Exchange分配到队列Queue。
   1. Direct-Exchange

    Direct Exchange是RabbitMQ Broker的默认Exchange,在此类型下,不必指定routing key的名字,创建的Queue有一个默认的routing key,一般与创建的Queue同名。
    

   2. Topic-Exchange

    Topic Exchange是根据routing key和Exchange的类型将message发送到一个或者多个Queue中,可以通过它来实现pub/sub模式,即发布订阅。
    

    注:“#”表示0个或若干个关键字,“*”表示一个关键字
    如果Exchange没有发现能够与RouteKey匹配的Queue,则会抛弃此消息

   3. Fanout-Exchange

    此类型的Exchange比较特殊,会忽略routing key的存在,直接将message广播到所有的Queue中。
    

   4. Headers-Exchange

    Headers Exchange不同于上面三种Exchange,它是根据Message的一些头部信息来分发过滤Message,忽略routing key的属性,如果Header信息和message消息的头信息相匹配,那么这条消息就匹配上了。
    

Spring集成RabbitMQ实现Topic模式

   1. rabbitmq.properties(mq配置文件)

rabbitmq.host=127.0.0.1
rabbitmq.port=5672
rabbitmq.username=admin
rabbitmq.password=admin
rabbitmq.my.exchange=my-exchange
rabbitmq.my.queue=my-queue

   2. 自动注入配置类(注意:需要创建自定义rabbitAdmin,否则启动时不会自动创建exchange、queue、bindings相关信息,具体可以查看RabbitAdmin类中的initialize()方法)    

/**
* @Description: rabbit mq相关配置
* @author yehaixiao
* @date 2018年5月30日
*/
@Configuration
public class RabbitMQConfig { private static final Logger LOGGER = Logger.getLogger(RabbitMQConfig.class); @Value("${rabbitmq.host}")
String host;
@Value("${rabbitmq.port}")
int port;
@Value("${rabbitmq.username}")
String username;
@Value("${rabbitmq.password}")
String password;
@Value("${rabbitmq.my.exchange}")
String exchange;
@Value("${rabbitmq.my.queue}")
String queue; /**
* 创建rabbit mq连接工厂
*/
@Bean(name = "rabbitConnectionFactory")
public ConnectionFactory connectionFactory() {
CachingConnectionFactory connectionFactory = new CachingConnectionFactory();
connectionFactory.setAddresses(host + ":" + port);
connectionFactory.setUsername(username);
connectionFactory.setPassword(password);
connectionFactory.setPublisherConfirms(true);
return connectionFactory;
} /**
* 自定义connectionFactory,需要声明rabbitAdmin 内部initialize()方法会进行exchanges、queues、bindings声明
*/
@Bean
public RabbitAdmin rabbitAdmin() {
return new RabbitAdmin(connectionFactory());
} /**
* 创建amqp消息模版,用于发送者发送消息
*/
@Bean
public AmqpTemplate amqpTemplate() {
RabbitTemplate amqpTemplate = new RabbitTemplate(connectionFactory());
// json消息转化
amqpTemplate.setMessageConverter(messageConverter());
return amqpTemplate;
} /**
* 创建topic类型交换器
*/
@Bean
public TopicExchange topicExchange() {
return new TopicExchange(exchange, false, true);
} /**
* 根据hostname创建queue(目的:为了实现多个集群中多个实例共享一条消息)
* durable:是否持久化
* exclusive:仅创建者可以使用的私有队列,断开后自动删除
* auto-delete:当所有消费端连接断开后,是否自动删除队列
*/
@Bean
public Queue queue() {
InetAddress netAddress = null;
try {
netAddress = InetAddress.getLocalHost();
} catch (UnknownHostException e) {
e.printStackTrace();
}
String hostName = netAddress.getHostName();
String queueName = String.format("%s." + queue, hostName);
LOGGER.info("dynamic create queue name:" + queueName);
Queue queue = new Queue(queueName, false, false, true);
return queue;
} /**
* 将队列queue与exchange绑定,binding_key为#.queue,模糊匹配
*/
@Bean
public Binding binding(Queue queue, TopicExchange topicExchange) {
return BindingBuilder.bind(queue).to(topicExchange).with("#." + queue);
} /**
* 创建消息监听器
*/
@Bean
public MessageListener customizeMessageListener() {
return new CustomizeMessageListener();
} /**
* 创建json消息转化器
*/
@Bean
public MessageConverter messageConverter() {
Jackson2JsonMessageConverter fastJsonMessageConverter = new Jackson2JsonMessageConverter();
return fastJsonMessageConverter;
} /**
* 绑定queue和listener关系
*/
@Bean
public SimpleMessageListenerContainer mqMessageContainer(Queue queue, MessageListener customizeMessageListener) {
SimpleMessageListenerContainer container = new SimpleMessageListenerContainer(connectionFactory());
container.setQueues(queue);
container.setExposeListenerChannel(true);
container.setAcknowledgeMode(AcknowledgeMode.AUTO);
container.setMessageListener(customizeMessageListener);
return container;
}
}

   3. 消息消费者(通过实现MessageListener来重写onMessage()方法)

/**
* @Description: 消息监听器
* @author yehaixiao
* @date 2018年5月24日
*/
@Service
public class CustomizeMessageListener implements MessageListener { private static final Logger LOGGER = LoggerFactory.getLogger(CustomizeMessageListener.class); @Override
public void onMessage(Message message) {
try {
String queue = message.getMessageProperties().getConsumerQueue();
String msg = new String(message.getBody());
LOGGER.info("consumer msg : {}, from queue : {}", msg, queue);
} catch (Exception e) {
LOGGER.error("consumer message error!", e);
}
}
}

   4. 消息生产者(通过AmqpTemplate发送消息)

/**
* @Description: 消息生产者
* @author yehaixiao
* @date 2018年5月28日
*/
@Service
public class MessageProducer { private static final Logger LOGGER = LoggerFactory.getLogger(MessageProducer.class); @Value("${rabbitmq.my.exchange}")
private String exchange; @Value("${rabbitmq.my.queue}")
private String queue; @Resource
private AmqpTemplate amqpTemplate; public void sendMessage(Object message) {
LOGGER.info("send message : {} , to queue : {}", JSON.toJSONString(message), queue);
// message是需要传递的信息, 指定特定的queue
amqpTemplate.convertAndSend(exchange, queue, message);
}
}

RabbitMQ Exchange详解以及Spring中Topic实战的更多相关文章

  1. .NET 云原生架构师训练营(模块二 基础巩固 RabbitMQ Masstransit 详解)--学习笔记

    2.6.7 RabbitMQ -- Masstransit 详解 Consumer 消费者 Producer 生产者 Request-Response 请求-响应 Consumer 消费者 在 Mas ...

  2. 由一个RABBITMQ监听器死循环引出的SPRING中BEAN和MAPPER接口的注入问题

    1 @Slf4j 2 @RestController 3 @Component 4 public class VouchersReceiverController implements Message ...

  3. 详解 $_SERVER 函数中QUERY_STRING和REQUEST_URI区别

    详解 $_SERVER 函数中QUERY_STRING和REQUEST_URI区别 http://blog.sina.com.cn/s/blog_686999de0100jgda.html   实例: ...

  4. 详解 Go 语言中的 time.Duration 类型

    swardsman详解 Go 语言中的 time.Duration 类型swardsman · 2018-03-17 23:10:54 · 5448 次点击 · 预计阅读时间 5 分钟 · 31分钟之 ...

  5. 详解jquery插件中(function ( $, window, document, undefined )的作用。

    1.(function(window,undefined){})(window); Q:(function(window,undefined){})(window);中为什么要将window和unde ...

  6. zz详解深度学习中的Normalization,BN/LN/WN

    详解深度学习中的Normalization,BN/LN/WN 讲得是相当之透彻清晰了 深度神经网络模型训练之难众所周知,其中一个重要的现象就是 Internal Covariate Shift. Ba ...

  7. [转载]详解网络传输中的三张表,MAC地址表、ARP缓存表以及路由表

    [转载]详解网络传输中的三张表,MAC地址表.ARP缓存表以及路由表 虽然学过了计算机网络,但是这部分还是有点乱.正好在网上看到了一篇文章,讲的很透彻,转载过来康康. 本文出自 "邓奇的Bl ...

  8. 详解WebService开发中四个常见问题(2)

    详解WebService开发中四个常见问题(2)   WebService开发中经常会碰到诸如WebService与方法重载.循环引用.数据被穿该等等问题.本文会给大家一些很好的解决方法. AD:WO ...

  9. 详解WebService开发中四个常见问题(1)

    详解WebService开发中四个常见问题(1)   WebService开发中经常会碰到诸如WebService与方法重载.循环引用.数据被穿该等等问题.本文会给大家一些很好的解决方法. AD:WO ...

随机推荐

  1. 字符编码中ASCII、Unicode和UTF-8的区别

    1. ASCII码 我们知道,在计算机内部,所有的信息最终都表示为一个二进制的字符串.每一个二进制位(bit)有0和1两种状态,因此八个二进制位就可以组合出256种状态,这被称为一个字节(byte). ...

  2. wincvs配置方法

    1.安装wincvs_中文1.3.exe 2.安装cvsnt-2.5.03.2260.msi  安装过程中选择complete选项 3.安装python221-setup.exe 安装完成后,先配置环 ...

  3. kbmmw 做REST 服务签名认证的一种方式

    一般对外提供提供REST 服务,由于信息安全的问题, 都要采用签名认证,今天简单说一下在KBMMW 中如何 实现简单的签名服务? 整个签名服务,模仿阿里大鱼的认证方式,大家可以根据实际情况自己修改. ...

  4. docker学习笔记-命令大全

    容器生命周期管理 • Run OPTIONS说明: • -a :显示所有的容器,包括未运行的. • -f :根据条件过滤显示的内容. • --format :指定返回值的模板文件. • -l :显示最 ...

  5. 带权单源最短路发[稠密图](Dijkstra)

    对于稠密图,采用邻接矩阵较为合适 所以我们先构建一个邻接矩阵 typedef int Vertex; typedef int WeightType; //图 typedef struct MyGrap ...

  6. PMP:3.项目经理角色

    成员角色:整合指挥者 在团队中的职责:负终责 知识技能:综合技能&沟通   定义: 职能经理专注于对某个职能领域或业务部门的管理监督. 运营经理负责保证业务运营的高效性. 项目经理是由执行组织 ...

  7. Linux基础理论

    本节内容 1.  Linux的安装及相关配置 2.  UNIX和Linux操作系统概述 3.  Linux命令及帮助 4.  目录结构 6.  用户.群组和权限 7.  用户.群组和权限的深入讨论 1 ...

  8. 背水一战 Windows 10 (67) - 控件(控件基类): DependencyObject - CoreDispatcher, 依赖属性的设置与获取, 依赖属性的变化回调

    [源码下载] 背水一战 Windows 10 (67) - 控件(控件基类): DependencyObject - CoreDispatcher, 依赖属性的设置与获取, 依赖属性的变化回调 作者: ...

  9. 任务调度及远端管理(基于Quartz.net)

    这篇文章我们来了解一些项目中的一个很重要的功能:任务调度 可能有些同学还不了解这个,其实简单点说任务调度与数据库中的Job是很相似的东西 只不过是运行的物理位置与管理方式有点不一样,从功能上来说我觉得 ...

  10. 笔记:Activity的启动过程

    Activity的创建特点 作为四大组件之一的Activity,它不像普通java对像那样,可以new出来,然后去使用.而是调用 startActivity()这样的方式启动.那么Android系统是 ...