RabbitMq案例实战:RabbitMQ的TTL以及死信队列_延迟付款,手动补偿

一、题目要求

基于RabbitMQ的TTL以及死信队列,使用SpringBoot实现延迟付款,手动补偿操作。

1、用户下单后展示等待付款页面

2、在页面上点击付款的按钮,如果不超时,则跳转到付款成功页面

3、如果超时,则跳转到用户历史账单中查看因付款超时而取消的订单。

二、思路分析

2.1 架构设计

image.png

2.2 前置准备

安装rabbitmq

  • 安装环境:

      1. 虚拟机软件:VMWare 15.1.0
      1. 操作系统:CentOS Linux release 7.7.1908
      1. Erlang:erlang-23.0.2-1.el7.x86_64
      1. RabbitMQ:rabbitmq-server-3.8.4-1.el7.noarch
  • RabbitMQ的安装需要首先安装Erlang,因为它是基于Erlang的VM运行的。

  • RabbitMQ需要的依赖:socat和logrotate,logrotate操作系统中已经存在了,只需要安装socat就可以了。

  • RabbitMQ与Erlang的兼容关系详见:https://www.rabbitmq.com/which-erlang.html

    • img
  • 1、安装依赖:

    • yum install socat -y
  • 2、安装Erlang

  • 3、安装RabbitMQ

  • 4、启用RabbitMQ的管理插件

    • rabbitmq-plugins enable rabbitmq_management
  • 5、开启RabbitMQ

    • systemctl start rabbitmq-server

    • 或者

    • rabbitmq-server

    • 或者后台启动

    • rabbitmq-server -detached

  • 6、添加用户

    • rabbitmqctl add_user root 123456
  • 7、给用户添加权限

    • 给root用户在虚拟主机"/"上的配置、写、读的权限

    • rabbitmqctl set_permissions root -p / "." "." ".*"

  • 8、给用户设置标签

    • rabbitmqctl set_user_tags root administrator

    • 用户的标签和权限:

    • img
  • 9、打开浏览器,访问http://<安装了CentOS的VMWare虚拟机IP地址>:15672

    • img
  • 10、使用刚才创建的用户登录:

    • img

2.3 使用延迟队列

在AMQP协议和RabbitMQ中都没有相关的规定和实现。不过,我们似乎可以借助rabbitmq中的“死信队列”来变相的实现。

可以使用rabbitmq_delayed_message_exchange插件实现。

需要在虚拟机运行如下命令

rabbitmq-plugins enable rabbitmq_management

2.4 源码分析

pom.xml

<dependencies>
 <dependency>
 <groupId>org.springframework.boot</groupId>
 <artifactId>spring-boot-starter-amqp</artifactId>
 </dependency>
 <dependency>
 <groupId>org.springframework.boot</groupId>
 <artifactId>spring-boot-starter-web</artifactId>
 </dependency>
 <dependency>
 <groupId>org.springframework.boot</groupId>
 <artifactId>spring-boot-starter-thymeleaf</artifactId>
 </dependency>
 <dependency>
 <groupId>org.springframework.boot</groupId>
 <artifactId>spring-boot-starter-test</artifactId>
 <scope>test</scope>
 </dependency>
 <dependency>
 <groupId>org.springframework.amqp</groupId>
 <artifactId>spring-rabbit-test</artifactId>
 <scope>test</scope>
 </dependency>
 <!--lombok工具-->
 <dependency>
 <groupId>org.projectlombok</groupId>
 <artifactId>lombok</artifactId>
 <version>1.18.4</version>
 <scope>provided</scope>
 </dependency>
 </dependencies>

配置文件application.properties

spring.application.name=payDemo
spring.rabbitmq.host=192.168.31.204
spring.rabbitmq.virtual-host=test
spring.rabbitmq.username=root
spring.rabbitmq.password=123456
spring.rabbitmq.port=5672

设置收到确认消息

spring.rabbitmq.publisher-confirm-type=correlated
spring.rabbitmq.publisher-returns=true
spring.rabbitmq.listener.direct.acknowledge-mode=manual

spring.thymeleaf.prefix=classpath:/templates/
spring.thymeleaf.check-template-location=true
spring.thymeleaf.suffix=.html</pre>

此处用的virtual-host是test ,也可用默认的spring.rabbitmq.virtual-host=/

配置类RabbitMqConfig.java

@Configuration
@EnableRabbit
@ComponentScan("com.idstaa")
public class RabbitMqConfig {
 @Bean
 /**
 * 订单消息队列
 */
 public Queue orderQueue() {
 return QueueBuilder.durable("q.order").build();
 }

 @Bean
 /**
 * 订单消息队列
 */
 public Queue ttlQueue() {
 Map<String, Object> args = new HashMap<>();
 args.put("x-message-ttl", 10000);
 args.put("x-dead-letter-exchange", "ex.dlx");
 args.put("x-dead-letter-routing-key", "key.dlx");
 return new Queue("ttl.order", true, false, false, args);
 }

 /**
 * 死信队列,用于取消用户订单
 */
 @Bean
 public Queue dlxQueue() {
 Map<String, Object> args = new HashMap<>();
 return new Queue("q.dlx", true, false, false, args);
 }

 /**
 * 订单交换器
 */
 @Bean
 public Exchange orderExchange() {
 Map<String, Object> args = new HashMap<>();
 DirectExchange exchange = new DirectExchange("ex.order",
 true,
 false,
 args);
 return exchange;
 }

 /**
 * ttl交换器
 */
 @Bean
 public Exchange ttlExchange() {
 Map<String, Object> args = new HashMap<>();
 DirectExchange exchange = new DirectExchange("ex.ttl",
 true,
 false,
 args);
 return exchange;
 }

 /**
 * 订单交换器
 */
 @Bean
 public Exchange dlxExchange() {
 Map<String, Object> args = new HashMap<>();
 DirectExchange exchange = new DirectExchange("ex.dlx",
 true,
 false,
 args);
 return exchange;
 }

 /**
 * 用于发送下单,做分布式事务的MQ
 */
 @Bean
 public Binding orderBinding() {
 return BindingBuilder.bind(orderQueue()).to(orderExchange()).with("key.order").noargs();
 }

 /**
 * 用于等待用户支付的延迟队列绑定
 */
 @Bean
 public Binding ttlBinding() {
 return BindingBuilder.bind(ttlQueue()).to(ttlExchange()).with("key.ttl").noargs();
 }


 /**
 * 用于支付超时取消用户订单的死信队列绑定
 */
 @Bean
 public Binding dlxBinding() {
 return BindingBuilder.bind(dlxQueue()).to(dlxExchange()).with("key.dlx").noargs();
 }

 @Bean
 public RabbitAdmin rabbitAdmin(ConnectionFactory connectionFactory){
 return new RabbitAdmin(connectionFactory);
 }

 @Bean(name="rabbitMessageListenerContainer")
 public DirectMessageListenerContainer listenerContainer(ConnectionFactory connectionFactory){
 DirectMessageListenerContainer container = new DirectMessageListenerContainer(connectionFactory);
 container.setAcknowledgeMode(AcknowledgeMode.MANUAL);
 container.setPrefetchCount(5);
 container.setConsumersPerQueue(5);
 container.setMessagesPerAck(1);

 ThreadPoolTaskExecutor taskExecutor = new ThreadPoolTaskExecutor();
 taskExecutor.setCorePoolSize(10);
 taskExecutor.setMaxPoolSize(20);
 // 设置改属性,灵活设置并发
 container.setTaskExecutor(taskExecutor);
 return container;
 }

 @Bean
 public MessageConverter messageConverter(){
 return new Jackson2JsonMessageConverter();
 }


}

订单控制器类OrderController.java

@Controller
public class OrderController {
 @Autowired
 private RabbitTemplate rabbitTemplate;

 @RequestMapping("/createOrder")
 public String createOrder(Model model) throws ExecutionException, InterruptedException {
 Order order = new Order();
 order.setOrderId(UUID.randomUUID().toString().substring(0,10));
 order.setStatus("待支付");
 order.setUserId("jianyi");
 OrderDetail detail = new OrderDetail();
 detail.setItemId(UUID.randomUUID().toString().substring(0,5));
 detail.setItemName("");
 detail.setItemPrice(100d);
 detail.setNum(2);
 ArrayList detailList = new ArrayList();
 detailList.add(detail);
 order.setDetail(detailList);
 CorrelationData correlationData = new CorrelationData();
 rabbitTemplate.convertAndSend("ex.order",
 "key.order",
 order,
 correlationData);
 CorrelationData.Confirm confirm = correlationData.getFuture().get();
 boolean ack = confirm.isAck();
 if(!ack){
 return "failOrder";
 }
 System.out.println("发送延迟取消信息,10s不支付就取消"+",当前时间"+ new SimpleDateFormat("yyyy-MM-dd hh:mm:ss").format(new Date()));
 rabbitTemplate.convertAndSend("ex.ttl","key.ttl",order.getOrderId());
 model.addAttribute("orderId",order.getOrderId());
 return "order";
 }

 @RequestMapping("/failOrder/{orderId}")
 public String failOrder(@PathVariable  String orderId,Model model) throws ExecutionException, InterruptedException {
 // 修改订单状态
 System.out.println(orderId);
 model.addAttribute("orderId",orderId);
 return "fail";
 }

 @RequestMapping("/pay")
 public String pay(String orderId,Model model) throws ExecutionException, InterruptedException {
 // 修改订单状态
 System.out.println(orderId+"订单状态为已支付");
 model.addAttribute("orderId",orderId);
 return "success";
 }

 @RequestMapping("/cancelOrderView")
 public String cancelOrderView(String orderId,Model model) throws ExecutionException, InterruptedException {
 // 修改订单状态
 System.out.println(orderId+"订单状态为已取消");
 model.addAttribute("orderId",orderId);
 return "cancelOrderView";
 }
}

订单实体类Order.java OrderDetail.java

@Data
public class Order {
 private String orderId;

 private String userId;

 private String status;

 private ArrayList<OrderDetail> detail;
}

@Data
public class OrderDetail {
 private String itemId;

 private String itemName;

 private double itemPrice;

 private int num;
}

静态页面order.html

<!DOCTYPE html>
<html lang="en" xmlns:th="http://www.thymeleaf.org">
<head>
 <meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
 <meta name="viewport" content="width=device-width, initial-scale=1,shrink-to-fit=no">
 <title>支付界面</title>
 <link th:href="@{/css/bootstrap.min.css}" rel="stylesheet">
 <link th:href="@{/css/signin.css}" rel="stylesheet">
 <script src="../static/js/jquery-1.10.2.js"></script>
</head>
<script language="javascript">
 var num = 9; //倒计时的秒数
 var URL = "/cancelOrderView?orderId=[[${orderId}]]";
 var id = window.setInterval('doUpdate()', 1000);
 function doUpdate() {
 document.getElementById('page_div').innerHTML = '支付时间还剩'+num+'秒' ;
 if(num == 0) {
 window.clearInterval(id);
 window.location = URL;
 }
 num --;
 }
</script>
<body>
<div  th:text="${orderId}"></div>
<div>订单创建成功</div>
<div class="time"> <p id="page_div">支付时间还剩10s,请尽快支付</p>
<a th:href="@{/pay/(orderId=${orderId})}">去支付</a>
</div>
</body>
</html>

其他静态页面

cancelOrderView.html

<!DOCTYPE html>
<html lang="en" xmlns:th="http://www.thymeleaf.org">
<head>
 <meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
 <meta name="viewport" content="width=device-width, initial-scale=1,shrink-to-fit=no">
 <title>取消订单</title>
 <link th:href="@{/css/bootstrap.min.css}" rel="stylesheet">
 <link th:href="@{/css/signin.css}" rel="stylesheet">
 <script src="../static/js/jquery-1.10.2.js"></script>
</head>
<body>
<div  th:text="${orderId}"></div>订单已取消
</body>
</html>

failOrder.html

<!DOCTYPE html>
<html lang="en">
<head>
 <meta charset="UTF-8">
 <title>Title</title>
</head>
<body>
 failorder
</body>
</html>

index.html

<!DOCTYPE html>
<html lang="en">
<head>
 <meta charset="UTF-8">
 <title>Title</title>
</head>
<body>
 <a href="/createOrder">下单</a>
</body>
</html>

success.html

<!DOCTYPE html>
<html lang="en" xmlns:th="http://www.thymeleaf.org">
<head>
 <meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
 <meta name="viewport" content="width=device-width, initial-scale=1,shrink-to-fit=no">
 <title>取消订单</title>
 <link th:href="@{/css/bootstrap.min.css}" rel="stylesheet">
 <link th:href="@{/css/signin.css}" rel="stylesheet">
 <script src="../static/js/jquery-1.10.2.js"></script>
</head>
<body>
<div  th:text="${orderId}"></div>支付成功
</body>
</html>

三、效果演示

前置准备,启动rabbitmq。设置Virtualhost = test 的虚机

1、启动spring项目。访问页面

image.png

2、点击下单按钮。不支付

image.png

image.png

页面倒计时,查看后台日志

image.png

10s后自动取消订单,并跳转订单取消页面

image.png

3、点击下单按钮。支付

image.png

查看后台日志

image.png

4、也可查看rabbitmq队列的变化,可自行查看

image.png

附件:github源码

https://github.com/xjdm/rabbitmq_paydemo
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,928评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,192评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,468评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,186评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,295评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,374评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,403评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,186评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,610评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,906评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,075评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,755评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,393评论 3 320
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,079评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,313评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,934评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,963评论 2 351