spring-boot-learning-RabbitMQ
为什么需要MQ???
异步处理:
场景:
用户注册后,需要发注册邮件和注册短信,传统的做法有两种 1.串行的方式 2.并行的方式`

用户注册-发送注册邮箱-发送注册短信都完成之后才结束返回给客户端
邮件,短信并不是必须的,它只是一个通知,而这种做法让客户端等待没有必要等待的东西.
耗时150ms
并行:

将注册信息写入数据库后,发送邮件的同时,发送短信,以上三个任务完成后,返回给客户端,并行的方式能提高处理的时间
耗时100ms
消息队列:
假设三个业务节点分别使用50ms,串行方式使用时间150ms,并行使用时间100ms。虽然并行已经提高的处理时间,但是,
前面说过,邮件和短信对我正常的使用网站没有任何影响,客户端没有必要等着其发送完成才显示注册成功,应该是写入
数据库后就返回.
`消息队列`: 引入消息队列后,把发送邮件,短信不是必须的业务逻辑异步处理
应用解耦:
场景:
双11是购物狂节,用户下单后,订单系统需要通知库存系统,传统的做法就是订单系统调用库存系统的接口

注意:当库存系统出现故障时,订单就会失败。订单系统和库存系统高耦合
引入消息队列

`订单系统:`用户下单后,订单系统完成持久化处理,将消息写入消息队列,返回用户订单下单成功。 库存系统:`订阅下单的消息,获取下单消息,进行库操作。 就算库存系统出现故障,消息队列也能保证消息的可靠投递,
不会导致消息丢失.
流量削峰:
场景:
秒杀活动,一般会因为流量过大,导致应用挂掉,为了解决这个问题,一般在应用前端加入消息队列。

1.用户的请求,服务器收到之后,首先写入消息队列,加入消息队列长度超过最大值,则直接抛弃用户请求或跳转到错误页面.
2.秒杀业务根据消息队列中的请求信息,再做后续处理.
起到的作用就是:
1.可以控制活动人数,超过此一定阀值的订单直接丢弃
2.可以缓解短时间的高流量压垮应用(应用程序按自己的最大处理能力获取订单)
springboot- 配置文件:
spring.application.name=springboot_rabbitmq
spring.rabbitmq.host=localhost
spring.rabbitmq.port=5672
spring.rabbitmq.username=quan
spring.rabbitmq.password=admin
spring.rabbitmq.virtual-host=quan
server.port=9898
RabbitMQ介绍
https://www.cnblogs.com/java-quan/p/13560266.html
第一种模型:直连
生产者:
@Autowired
RabbitTemplate template; @Test
void producer1() {
template.convertAndSend("hello","hello-rabbit");
}
消费者:
@Component
@RabbitListener(queuesToDeclare = @Queue("hello"))
public class Customer {
@RabbitHandler
public void remsg(String msg){
System.out.println("message== "+msg);
}
}
第二种模型:任务模型(work quene)
producer;
@Autowired
RabbitTemplate template; @Test
void producer2() {
for (int i = 0;i<10;i++){
template.convertAndSend("work",i+"hello-rabbit");
System.out.println(i);
}
customer:
@Component
public class Customer2 {
@RabbitListener(queuesToDeclare = @Queue("work"))
public void receive1(String msg){
System.out.println("work-msg1= " +msg);
} @RabbitListener(queuesToDeclare = @Queue("work"))
public void receive2(String msg){
System.out.println("work-msg2= " +msg);
}
}
结果:
* work-msg1= 1hello-rabbit
* work-msg2= 0hello-rabbit
* work-msg1= 5hello-rabbit
* work-msg2= 4hello-rabbit
* work-msg1= 9hello-rabbit
* work-msg2= 8hello-rabbit
默认在Spring AMQP实现中Work这种方式就是公平调度,如果需要实现能者多劳需要额外配置`
第三种模型--广播
producer:
@Autowired
RabbitTemplate template; @Test
void producer3() {
template.convertAndSend("logs2","","广播广播");
}
customer:
@Component
public class Customer3 {
@RabbitListener(bindings = @QueueBinding(
value = @Queue,
exchange = @Exchange(name = "logs2",type = "fanout")
)
)
public void receive1(String msg){
System.out.println("msg1= " +msg);
} @RabbitListener(bindings = @QueueBinding(
value = @Queue,//c创造临时队列
exchange = @Exchange(name = "logs2",type = "fanout")//绑定交换机类型
)
)
public void receive2(String msg){
System.out.println("msg2= " +msg);
}
}
结果:
// msg2= 广播广播
// msg1= 广播广播
第四种模式:Routing-订阅模式中的-直连(direct)
producer:
@Autowired
RabbitTemplate template; @Test
void producer4() {
template.convertAndSend("directs","error","广播广播");
template.convertAndSend("directs","info","广播广播");
template.convertAndSend("directs","debug","广播广播");
}
customer:
@Component
public class Customer4 {
@RabbitListener(bindings = @QueueBinding(
value = @Queue,
key = {"info","error","debug"},
exchange = @Exchange(name = "directs",type = "direct")
)
)
public void receive1(String msg){
System.out.println("msg1-info-error-debug= " +msg);
} @RabbitListener(bindings = @QueueBinding(
value = @Queue,//c创造临时队列
key = {"info"},
exchange = @Exchange(name = "directs",type = "direct")//绑定交换机类型
)
)
public void receive2(String msg){
System.out.println("msg2-info= " +msg);
} @RabbitListener(bindings = @QueueBinding(
value = @Queue,//c创造临时队列
key = {"error"},
exchange = @Exchange(name = "directs",type = "direct")//绑定交换机类型
)
)
public void receive3(String msg){
System.out.println("msg3-error= " +msg);
} }
结果:
/**
* msg3-error= 广播广播
* msg2-info= 广播广播
* msg1-info-error-debug= 广播广播
* msg1-info-error-debug= 广播广播
* msg1-info-error-debug= 广播广播
*/
Routing的订阅模式--topic:
producer
@Test
void producer5() {
template.convertAndSend("topics", "user.quan", "user.quan广播广播");
template.convertAndSend("topics", "user.quan.all", "user.quan.all广播广播"); }
customer
@Component
public class Customer5 {
@RabbitListener(bindings = @QueueBinding(
value = @Queue,
key = {"user.*"},
exchange = @Exchange(name = "topics",type = "topic")
)
)
public void receive1(String msg){
System.out.println("msg1-info-error-debug= " +msg);
} @RabbitListener(bindings = @QueueBinding(
value = @Queue,//c创造临时队列
key = {"user.#"},
exchange = @Exchange(name = "topics",type = "topic")//绑定交换机类型
)
)
public void receive2(String msg){
System.out.println("msg2-info= " +msg);
} }
结果:
/**
* msg2-info= user.quan广播广播
* msg1-info-error-debug= user.quan广播广播
* msg2-info= user.quan.all广播广播
*/
spring-boot-learning-RabbitMQ的更多相关文章
- 85. Spring Boot集成RabbitMQ【从零开始学Spring Boot】
这一节我们介绍下Spring Boot整合RabbitMQ,对于RabbitMQ这里不过多的介绍,大家可以参考网络上的资源进行安装配置,本节重点是告诉大家如何在Spring Boot中使用Rabbit ...
- Spring Boot (十三): Spring Boot 整合 RabbitMQ
1. 前言 RabbitMQ 是一个消息队列,说到消息队列,大家可能多多少少有听过,它主要的功能是用来实现应用服务的异步与解耦,同时也能起到削峰填谷.消息分发的作用. 消息队列在比较主要的一个作用是用 ...
- Spring Boot 集成 RabbitMQ 实战
Spring Boot 集成 RabbitMQ 实战 特别说明: 本文主要参考了程序员 DD 的博客文章<Spring Boot中使用RabbitMQ>,在此向原作者表示感谢. Mac 上 ...
- Spring Boot 整合 rabbitmq
一.消息中间件的应用场景 异步处理 场景:用户注册,信息写入数据库后,需要给用户发送注册成功的邮件,再发送注册成功的邮件. 1.同步调用:注册成功后,顺序执行发送邮件方法,发送短信方法,最后响应用户 ...
- Spring Boot 实现 RabbitMQ 延迟消费和延迟重试队列
本文主要摘录自:详细介绍Spring Boot + RabbitMQ实现延迟队列 并增加了自己的一些理解,记录下来,以便日后查阅. 项目源码: spring-boot-rabbitmq-delay-q ...
- spring boot整合RabbitMQ(Direct模式)
springboot集成RabbitMQ非常简单,如果只是简单的使用配置非常少,springboot提供了spring-boot-starter-amqp项目对消息各种支持. Direct Excha ...
- spring boot集成RabbitMQ
原文:https://www.jianshu.com/p/e1258c004314 RabbitMQ作为AMQP的代表性产品,在项目中大量使用.结合现在主流的spring boot,极大简化了开发过程 ...
- Spring boot集成RabbitMQ(山东数漫江湖)
RabbitMQ简介 RabbitMQ是一个在AMQP基础上完整的,可复用的企业消息系统 MQ全称为Message Queue, 消息队列(MQ)是一种应用程序对应用程序的通信方法.应用程序通过读写出 ...
- Spring Boot 集成RabbitMQ
在Spring Boot中整合RabbitMQ是非常容易的,通过在Spring Boot应用中整合RabbitMQ,实现一个简单的发送.接收消息的例子. 首先需要启动RabbitMQ服务,并且add一 ...
- Spring Boot整合Rabbitmq
Spring Boot应用中整合RabbitMQ,并实现一个简单的发送.接收消息的例子来对RabbitMQ有一个直观的感受和理解. 在Spring Boot中整合RabbitMQ是一件非常容易的事,因 ...
随机推荐
- Pandas:DataFrame绘制并保存折线图时不打开图形只保存文件
保存图形,用的是plt.savefig函数,只需要在保存图形之后,调用plt.close()关闭画布,就不会显示出来了: data.plot() outfile='image.png' plt.sav ...
- Qt:打印输出到控制台,类似C++的cout
1. #include<qDebug> 2. qDebug<<"Hello,world!"; 补充,如果不是控制台文件,比如是窗口应用程序,需要在pro文件 ...
- urllib-访问网页的两种方式:GET与POST
学习自:https://www.jianshu.com/p/4c3e228940c8 使用参数.关键字访问服务器 访问网络的两种方法: 1.GET 利用参数给服务器传递信息 参数data为dict类型 ...
- C# Winform中FpSpread表格控件设置固定的(冻结的)行或列
在项目中我们经常会用到固定表头的操作,FpSpread提供了冻结行或列的属性. 你可以冻结表单中的行或列(使其不可滚动). 你可以冻结任意个表单顶部的行,使其成为前导行,你也可以冻结左侧任意多个列,使 ...
- Python脚本生成可执行文件&(恋爱小脚本)
Python脚本生成可执行文件&(恋爱小脚本) 参考文献: http://c.biancheng.net/view/2690.html; https://blog.csdn.net/qq_39 ...
- Ajax 实战(一)
目录 Ajax 实战(一) 简介 入门案例 基于Ajax进行登录验证 HTTP请求编码格式和报文 Content-Type=application/x-www-form-urlencoded Cont ...
- Tableau绘图一热图、日历图、人口金字塔、标靶图、凹凸图、帕累托图
Tableau绘图一热图.日历图.人口金字塔.标靶图.凹凸图.帕累托图 本文首发于博客冰山一树Sankey,去博客浏览效果更好.直接右上角搜索该标题即可 一.热图 例子:示例超市 可以通过更改颜色来改 ...
- laravel 访问器 和修改器的使用
对于访问器我是这样定义的,就是将数据库中的数据被访问时可以变成我们想要的数据类型(例如:数据库中的时间字段是int类型,要将她变成data(Y-m-d H:i:s),格式类型) 参看博客 https: ...
- oop简易封装增删改查
//注意要先引入含有封装类的文件文件:如下: <?phpclass Db{ public $host='127.0.0.1'; public $user='root'; public $pass ...
- PhpStrom 好用的翻译插件
最近php使用laravel框架的比较多,里面的注释都是英文的,有些同学的英语不是很好,不过不用但是侯蜀黍带你一个好用的翻译插件,告别烦恼一了百了 Translation 翻译插件 安装: 打开Fil ...