使用rabbitmq实现集群im聊天服务器消息的路由
这个地址图文会更清晰:https://www.jianshu.com/p/537e87c64ac7
单机系统的时候,客户端和连接都有同一台服务器管理。

在本地维护一份userId到connetciont的映射
服务器可以根据userId找出对应的连接,然后把消息push出去

但是集群环境下,连接分布在不同的机器,小明向小张发消息时跨了机器

小明向小张发的消息,需要小张的对应连接的服务器才能推送
要完成这个需求需要解决两个问题:
1、聊天服务器这么多,怎么才能知道小张连接了哪一台机器?
2、知道是哪一台服务器,怎么才能把消息精准发送到对应的机器?

有一种解决方案,就是把消息全量广播,然后每台机器自己去判断小张是不是在本机有连接,然后对应推送消息。但是这个方案明显不可取,因为这样会造成网络风暴,满天飞羽的消息在广播。
个人解决方案
一、对于第一个问题,怎么才能知道小张连接了哪一台机器,可以用redis去做一个map映射。
每台机器启动的时候都分配一个机器id,
当用户创建连接的时候,在redis放一个key-value,key值放的是userId,value放的是机器id。
这样就可以根据userId找对应的连接机器

二、对于第二个问题,怎么才能把消息精准发送到对应的机器,可以让每台机器启动的时候都有一个专属的mq和专属的routingkey,使用rabbitMq的TopicExchange交换器,这样消息就能精准投递到对应的机器,routingKey可以用上面定义的机器id。
同时queue的熟悉选专属队列,这样服务器重启后,连接断开后,旧的队列会自动删除

全过程就是:
1、小明要发消息给小张
2、根据小张userId找到小张所在的机器b
3、构建messageBody,把routingKey设置成b
4、向mq发送消息,rabbitMq根据routingKey把消息分发给机器b;
以下是代码:
1、新建springboot工程,maven如下
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-amqp</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
<optional>true</optional>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
<scope>test</scope>
<exclusions>
<exclusion>
<groupId>org.junit.vintage</groupId>
<artifactId>junit-vintage-engine</artifactId>
</exclusion>
</exclusions>
</dependency>
<dependency>
<groupId>org.springframework.amqp</groupId>
<artifactId>spring-rabbit-test</artifactId>
<scope>test</scope>
</dependency>
</dependencies>
2、新建Constant做固定配置
public interface Constant {
//队列名称,这个用uuid
String QUEUE = "queue:"+UUID.randomUUID().toString();
//路由键,作为机器id
String ROUTING_KEY = "routingKey:"+new Date().getTime();
//固定值,代表消息主题
String TOPIC = "topic";
//redsikey的前缀
String REDIS_KEY = "redisKey:";
}
3、配置RabbitmqConfig
@Configuration
public class RabbitmqConfig {
//新建topic交换器
@Bean
TopicExchange initExchange() {
return new TopicExchange(Constant.TOPIC);
}
//新建队列,要设置独占队列,exclusive
@Bean
public Queue initQueue() {
Queue queue = QueueBuilder.durable(Constant.QUEUE).exclusive().autoDelete().build();
return queue;
}
//交换器和主题绑定
@Bean
Binding bindingiTopicExchange() {
return BindingBuilder.bind(initQueue()).to(initExchange()).with(Constant.ROUTING_KEY);
}
//新建消费者
@Bean
public SimpleMessageListenerContainer container(ConnectionFactory connectionFactory, ChannelAwareMessageListener listener) {
SimpleMessageListenerContainer container = new SimpleMessageListenerContainer();
container.setConnectionFactory(connectionFactory);
// 指定消费者
container.setMessageListener(listener);
// 指定监听的队列
container.setQueueNames(Constant.QUEUE);
// 设置消费者的 ack 模式为手动确认模式
container.setAcknowledgeMode(AcknowledgeMode.MANUAL);
container.setPrefetchCount(300);
//connection
return container;
}
}
4、配置消费者Consumer
@Slf4j
@Component
public class Consumer implements ChannelAwareMessageListener {
@Autowired
private RabbitTemplate rabbitTemplate;
@Override
public void onMessage(Message message, Channel channel) throws Exception {
MessageProperties messageProperties = message.getMessageProperties();
// 代表投递的标识符,唯一标识了当前信道上的投递,通过 deliveryTag ,消费者就可以告诉 RabbitMQ 确认收到了当前消息,见下面的方法
long deliveryTag = messageProperties.getDeliveryTag();
// 如果是重复投递的消息,redelivered 为 true
Boolean redelivered = messageProperties.getRedelivered();
// 获取生产者发送的原始消息
Object originalMessage = rabbitTemplate.getMessageConverter().fromMessage(message);
log.info("接受到消息:{}",originalMessage);
// 代表消费者确认收到当前消息,第二个参数表示一次是否 ack 多条消息
channel.basicAck(deliveryTag, false);
}
}
5、设置UserController
@Slf4j
@RequestMapping
@RestController
public class UserController {
@Autowired
private RedisTemplate redisTemplate;
@Autowired
private RabbitTemplate rabbitTemplate;
@GetMapping("/sendMsg")
public String sendMsg(@RequestParam Integer sendUserId,@RequestParam Integer recUserId,String msg ){
MessageProperties messageProperties = new MessageProperties();
String messageBody = msg;
Message message = rabbitTemplate.getMessageConverter().toMessage(messageBody, messageProperties);
String key = Constant.REDIS_KEY+recUserId;
Object o = redisTemplate.opsForValue().get(key);
if(o == null){
throw new RuntimeException("recUserId未建立连接:"+recUserId);
}
rabbitTemplate.convertAndSend(Constant.TOPIC, o.toString(),message);
return "success";
}
@GetMapping("/buildRoutingKey")
public String buildRoutingKey(@RequestParam Integer recUserId){
String key = Constant.REDIS_KEY+recUserId;
log.info("key={},value={}",key,Constant.ROUTING_KEY);
redisTemplate.opsForValue().set(key,Constant.ROUTING_KEY);
return "success";
}
}
6、实验环节,需要部署两个jar包
分别在2台服务器发起请求初始化连接。
在A服务器输入 curl localhost:8081//buildRoutingKey?recUserId=1
在B服务器输入 curl localhost:8082//buildRoutingKey?recUserId=2
然后发送消息
在任意一台服务器输入 curl localhost:8082/sendMsg?sendUserId=2&recUserId=1&msg=xiaozhangHello
在任意一台服务器输入 curl localhost:8082/sendMsg?sendUserId=1&recUserId=2&msg=xiaoMingHello


代码地址
https://github.com/hd-eujian/rabbitmq-share.git
https://gitee.com/guoeryyj/rabbitmq-share.git
使用rabbitmq实现集群im聊天服务器消息的路由的更多相关文章
- 关于RabbitMQ分布式集群架构
RabbitMQ分布式集群架构和高可用性(HA) (一) 功能和原理 设计集群的目的 允许消费者和生产者在RabbitMQ节点崩溃的情况下继续运行 通过增加更多的节点来扩展消息通信的吞吐量 1 集群配 ...
- RabbitMQ之集群搭建
1.RabbitMQ集群模式RabbitMQ集群中节点包括内存节点(RAM).磁盘节点(Disk,消息持久化),集群中至少有一个Disk节点. 2.普通模式(默认) 对于普通模式,集群中 ...
- RabbitMQ入门教程(十四):RabbitMQ单机集群搭建
原文:RabbitMQ入门教程(十四):RabbitMQ单机集群搭建 版权声明:本文为博主原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接和本声明. 本文链接:https://b ...
- 11.RabbitMQ单机集群
RabbitMQ集群设计用于完成两个目标:允许消费者和生产者在RabbitMQ节点崩溃的情况下继续运行,以及通过添加更多的节点来扩展消息通信的吞吐量. RabbitMQ会始终记录以下四种类型的内部元数 ...
- Rabbitmq关于集群节点功能的读书笔记
消息和队列可以指定是否持久化,如果指定持久化则会保存到硬盘上 ,不然只在内存里 普通集群模式下持久化的队列不能重建了 内存节点和磁盘节点的区别就是将元数据放在了内存还是硬盘,仅此而已,当在集群中声明队 ...
- 2) broadcast,这是启动完毕之后,集群中的服务器开始接收客户端的连接一起工作的过程,如果客户端有修改数据的改动,那么一定会由leader广播给follower,所以称为”broadcast”.
2) broadcast,这是启动完毕之后,集群中的服务器开始接收客户端的连接一起工作的过程,如果客户端有修改数据的改动,那么一定会由leader广播给follower,所以称为”broadcast” ...
- zookeeper集群,每个服务器上的数据是相同的,每一个服务器均可以对外提供读和写的服务,这点和redis是相同的,即对客户端来讲每个服务器都是平等的。
zookeeper集群,每个服务器上的数据是相同的,每一个服务器均可以对外提供读和写的服务,这点和redis是相同的,即对客户端来讲每个服务器都是平等的.
- RabbitMQ (十三) 集群+单机搭建(window)
拜读了网上很多前辈的文章,对RabbitMQ的集群有了一点点认识. 好多文章都说到,RabbitMQ的集群分为普通集群和镜像集群,有的还加了两种:单机集群和主从集群. 我看来看去,看了半天,怎么感觉, ...
- 部署rabbitMQ镜像集群实战测试
部署rabbitMQ镜像集群 版本信息 rabbit MQ: 3.8.5 Erlang: 官方建议最低21.3 推荐22.x 这里用的是23 环境准备 主机规划 主机 节点 172.16.14.3 磁 ...
随机推荐
- 027 01 Android 零基础入门 01 Java基础语法 03 Java运算符 07 逻辑“与”运算符
027 01 Android 零基础入门 01 Java基础语法 03 Java运算符 07 逻辑"与"运算符 本文知识点:Java中的逻辑"与"运算符 逻辑运 ...
- python数据结构之图深度优先和广度优先实例详解
本文实例讲述了python数据结构之图深度优先和广度优先用法.分享给大家供大家参考.具体如下: 首先有一个概念:回溯 回溯法(探索与回溯法)是一种选优搜索法,按选优条件向前搜索,以达到目标.但当探索到 ...
- python中numpy.savetxt 参数
转载:https://blog.csdn.net/qq_36535820/article/details/99543188 numpy.savetxt 参数 numpy.savetxt(fname,X ...
- Matlab中num2str函数的用法
转载:https://blog.csdn.net/SMF0504/article/details/51836062 函数功能: 把数值转换成字符串, 转换后可以使用fprintf或disp函数进行输出 ...
- python中input()函数与print()函数
一.input()函数详解 二.print()函数详解 三.类型转换
- day64 Pyhton 框架Django 07
day67 内容回顾 视图 1. CBV 定义 from django.views import View class Addpub(View): def get(self,request): sel ...
- day21 Pyhton学习 模块
一.模块:就是一个包含了python定义和声明的文件,文件名是模块名字加上.py的后缀. 但其实import加载的模块分为四个通用类别: 1.使用python编写的代码(.py文件) 2.已被编译为共 ...
- js产生任意2个区间内的随机整数
var code = myRound(30,100); function myRound(begin,end){ var num = Math.round(Math.random()*(end-beg ...
- win32获取进程树,以及命令行参数
1.先上代码 package main import ( "bytes" "errors" "flag" "fmt" & ...
- 机器分配----线性dp难题(对于我来说)
题目: 总公司拥有高效设备M台, 准备分给下属的N个分公司.各分公司若获得这些设备,可以为国家提供一定的盈利.问:如何分配这M台设备才能使国家得到的盈利最大?求出最大盈利值.其中M <= 15, ...