RabbitMQ学习笔记(总结)

本文深入探讨RabbitMQ的使用,包括其在系统设计中的作用,如可扩展性和可集群性。解释了RabbitMQ的Exchange、RoutingKey和Queue的概念,并阐述了为何选择RabbitMQ,主要优点在于解耦、异步处理和流量削峰。此外,还讨论了RoutingKey在消息路由中的关键角色。
摘要由CSDN通过智能技术生成

点击上方蓝字"优派编程"选择“加为星标”,第一时间关注原创干货

一、RabbitMQ的深入理解和最简单的用途说明

如何在新的系统中使用RabbitMQ.

系统设计的两个重大问题.

第一条要满足未来的业务需求的不断变化和增加. 也就是可扩展性.

第二条要满足性能的可伸缩性. 也就是可集群性…通过增加机器能处理更多的请求

第三条要解耦合.

如果不解耦合, 未来业务增加或变更的时候你还在修改3年前写的代码.试问你有多大的把握保证升级好系统不出问题? 如何可以写新的代码而不用修改老代码所带来的好处谁都知道…

第四条简单易懂.

以上4条在任何一个系统中都要遵循的原则. 以前是无法做到的. 自从有了MQ以后. 这些都可以同时做到了.

以前的设计理念是把系统看作一个人,按照工作的指令从上到下的执行.

现在要建立的概念是, 把系统的各个功能看作不同的人. 人与人之间的沟通通过消息进行交流传递信息…

有了MQ以后把一个人的事情分给了不同的人, 分工合作所带来的好处是专业化, 并行化. 当然也引入了一些麻烦,性能开销多一些, 工作任务的完整性不能立即得到反馈.幸好我们可以通过最终一致性.来解决这个麻烦的问题…

第一个问题RabbitMQ是如何支持可扩展性的.

990b9cac800237de9bb89db66e84f9e8.png

第二个问题RabbitMQ如何满足性能的可伸缩性. 也就是可集群性

50cb0b308050dc85e4a60f34aed2f24f.png

如果订阅人的down机了. 信息会丢失吗?

事实上是不会的. 只要有邮箱(队列Queue)存在.信息就一直存在, 除非订阅人去取走.

如果订阅人一直down机, 邮箱队列能存多少信息?会不会爆掉?

理论上和实际上都是有上限的不可能无限多. 具体多少看硬盘吧..我没测到过上限.

我这篇文章并不打算讲解邮局的4种投递模式. 有其它文章讲的很好. 我只打算使用topic这种模式. 因为它更灵活一些.

再说一下我的另外两个观点.

不要在业务程序中用代码定义创建 邮局 ExChange. 和邮箱Queue队列 这属于系统设计者要构架的事情. 要有专门独立的程序和规则去创建. 这样可以统一管理事件类型.避免过多的乱七八糟的RoutingKey混乱.

我的理解认为

消息系统的分布式可扩展的实现在于消息广播, 集群性的实现在于邮箱队列.

RabbitMQ是先广播后队列的.

Exchange: 就是邮局的概念等同于 中国邮政和顺丰快递、

routingkey: 就是邮件地址的概念.

queue: 就是邮箱接收软件,但是可以接收多个地址的邮件,通过bind实现。

producer:消息生产者,就是投递消息的程序。

consumer:消息消费者,就是接受消息的程序。

channel:消息通道,在客户端的每个连接里,可建立多个channel,每个channel代表一个会话任务。


二、为什么要使用RabbitMQ?RabbitMQ有什么优点?

原因有三:

1.解耦

639b94258da83d8347d55c89004a892f.png

2.异步

2c6c22343e846f6125cb680609a521ad.png
3.削峰

e796de3a6ec4875e7b72ead98136799e.png


三、rabbitmq的routingkey的作用

对于消息发布者而言它只负责把消息发布出去,甚至它也不知道消息是发到哪个queue,消息通过exchange到达queue,exchange的职责非常简单,就是一边接收发布者的消息一边把这些消息推到queue中。

而exchange是怎么知道消息应该推到哪个queue呢,这就要通过绑定queue与exchange时的routingkey了,通过代码进行绑定并且指定routingkey,下面有一张关系图,p(发布者) —> x(exchange) bindding(绑定关系也就是我们的routingkey) 红色代表着queue

我们来看代码:

在消息的生产者端:

@Component
public class RabbitOrderSender {
//自动注入RabbitTemplate模板类
@Autowired
private RabbitTemplate rabbitTemplate;

@Autowired
private BrokerMessageLogMapper brokerMessageLogMapper;

//回调函数: confirm确认
final RabbitTemplate.ConfirmCallback confirmCallback = new RabbitTemplate.ConfirmCallback() {
@Override
public void confirm(CorrelationData correlationData, boolean ack, String cause) {
System.err.println("correlationData: " + correlationData);
String messageId = correlationData.getId();
if(ack){
//如果confirm返回成功 则进行更新
brokerMessageLogMapper.changeBrokerMessageLogStatus(messageId, Constants.ORDER_SEND_SUCCESS, new Date());
} else {
//失败则进行具体的后续操作:重试 或者补偿等手段
System.err.println("异常处理...");
}
}
};

//发送消息方法调用: 构建自定义对象消息
public void sendOrder(Order order) throws Exception {
// 通过实现 ConfirmCallback 接口,消息发送到 Broker 后触发回调,确认消息是否到达 Broker 服务器,也就是只确认是否正确到达 Exchange 中
rabbitTemplate.setConfirmCallback(confirmCallback);
//消息唯一ID
CorrelationData correlationData = new CorrelationData(order.getMessageId());
rabbitTemplate.convertAndSend("order-exchange", "order.ABC", order, correlationData);
}
}

利用rabbitTemplate(import org.springframework.amqp.rabbit.core.RabbitTemplate;需要在pom.xml中导入amqp的依赖)的convertAndSend方法就可以发送,这里order-exchange为交换机exchange,order.ABC为routingKey,并没有指定对应消息需要发往哪个队列,还有指定消息回调。

在消息的消费者端:

@Component
public class OrderReceiver {
//配置监听的哪一个队列,同时在没有queue和exchange的情况下会去创建并建立绑定关系
@RabbitListener(bindings = @QueueBinding(
value = @Queue(value = "order-queue",durable = "true"),
exchange = @Exchange(name="order-exchange",durable = "true",type = "topic"),
key = "order.*"
)
)
@RabbitHandler//如果有消息过来,在消费的时候调用这个方法
public void onOrderMessage(@Payload Order order, @Headers Map<String,Object> headers, Channel channel) throws IOException {
//消费者操作
System.out.println("---------收到消息,开始消费---------");
System.out.println("订单ID:"+order.getId());

/**
* Delivery Tag 用来标识信道中投递的消息。RabbitMQ 推送消息给 Consumer 时,会附带一个 Delivery Tag,
* 以便 Consumer 可以在消息确认时告诉 RabbitMQ 到底是哪条消息被确认了。
* RabbitMQ 保证在每个信道中,每条消息的 Delivery Tag 从 1 开始递增。
*/
Long deliveryTag = (Long) headers.get(AmqpHeaders.DELIVERY_TAG);

/**
* multiple 取值为 false 时,表示通知 RabbitMQ 当前消息被确认
* 如果为 true,则额外将比第一个参数指定的 delivery tag 小的消息一并确认
*/
boolean multiple = false;

//ACK,确认一条消息已经被消费。不然的话,在rabbitmq首页会有Unacked显示为未处理数1.
channel.basicAck(deliveryTag,multiple);
}
}

原文地址:http://wp.fang1688.cn/java/460.html

往期推荐

编程学习心得

2021年编程课程网盘资料大汇总,限时领取!

2021最新小程序云开发学习资源汇总

wordpress添加百度Ping加快百度收录

最新php开发学习资料网盘下载

最新安卓开发学习资料网盘下载


最近小编方包开发了一款工具类实用的小程序!功能包括:抖音去水印,天气预报,小说在线免费阅读(内含上万部热门小说),历史今天,生成图片二维码,图片识别文字,ai伪原创文章,文字转语音MP3功能...

点击下方【方包工具箱】小程序,开箱即用,所有工具免费无广告!

方包工具箱小程序(点击此处跳转)

eeabdd1d671b517aaa0713f90f6dfa76.png


送福利!关注下方的公众号:“优派编程”,搜索关键词“资料”,即可获得软件app下载资源和python、java等编程学习资料!

   4fd0ed00b16b93b97adea808632afb16.png

点击左下方“阅读原文”关注方包it博客:

https://www.fang1688.cn

领取更多福利!由于小编的公众号没有评论功能,如有疑问要交流可以点击左下方“阅读原文”进行评论留言,留下您的宝贵意见!

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值