在支付系统中,订单通常是具有时效性的,例如在下单30分钟后如果还没有完成支付,那么就要取消订单,不能再执行后续流程。说到这,可能大家的第一反应是启动一个定时任务,来轮询订单的状态是否完成了支付,如果超时还没有完成,那么就去修改订单的关闭字段。当然,在数据量小的时候这么干没什么问题,但是如果订单的数量上来了,那么就会出现读取数据的瓶颈,毕竟来一次全表扫描还是挺费时的。

针对于定时任务的这种缺陷,关闭订单的这个需求大多依赖于延时任务来实现,这里说明一下延时任务与定时任务的最大不同,定时任务有执行周期的,而延时任务在某事件触发后一段时间内执行,并没有执行周期。

对于延时任务,可能大家对于RabbitMQ的延时队列会比较熟悉,用起来也是得心应手,但是你是否知道使用Redis也能实现延时任务的功能呢,今天我们就来看看具体应该如何实现。

使用Redis实现的延时队列,需要借助Redisson的依赖:

<dependency><groupId>org.redisson</groupId><artifactId>redisson-spring-boot-starter</artifactId><version>3.10.7</version>
</dependency>

首先实现往延时队列中添加任务的方法,为了测试时方便,我们把延迟时间设为30秒。

@Component
public class UnpaidOrderQueue {@AutowiredRedissonClient redissonClient;public void addUnpaid(String orderId){RBlockingQueue<String> blockingFairQueue = redissonClient.getBlockingQueue("orderQueue");RDelayedQueue<String> delayedQueue = redissonClient.getDelayedQueue(blockingFairQueue);System.out.println(DateTime.now().toString(JodaUtil.HH_MM_SS)+" 添加任务到延时队列");delayedQueue.offer(orderId,30, TimeUnit.SECONDS);}
}

添加一个对队列的监听方法,通过实现CommandLineRunner接口,使它在springboot启动时就开始执行:

@Component
public class QueueRunner implements CommandLineRunner {@Autowiredprivate RedissonClient redissonClient;@Autowiredprivate OrderService orderService;@Overridepublic void run(String... args) throws Exception {new Thread(()->{RBlockingQueue<String> blockingFairQueue = redissonClient.getBlockingQueue("orderQueue");RDelayedQueue<String> delayedQueue = redissonClient.getDelayedQueue(blockingFairQueue);delayedQueue.offer(null, 1, TimeUnit.SECONDS);while (true){String orderId = null;try {orderId = blockingFairQueue.take();} catch (Exception e) {continue;}if (orderId==null) {continue;}System.out.println(String.format(DateTime.now().toString(JodaUtil.HH_MM_SS)+" 延时队列收到:"+orderId));System.out.println(DateTime.now().toString(JodaUtil.HH_MM_SS)+" 检测订单是否完成支付");if (orderService.isTimeOut(orderId)) {orderService.closeOrder(orderId);}}}).start();}
}

在方法中,单独启动了一个线程来进行监听,如果有任务进入延时队列,那么取到订单号后,调用我们OrderService提供的检测是否订单过期的服务,如果过期,那么执行关闭订单的操作。

创建简单的OrderService用于测试,提供创建订单,检测超时,关闭订单方法:

@Service
public class OrderService {@AutowiredUnpaidOrderQueue unpaidOrderQueue;public void createOrder(String order){System.out.println(DateTime.now().toString(JodaUtil.HH_MM_SS)+" 创建订单:"+order);unpaidOrderQueue.addUnpaid(order);}public boolean isTimeOut(String orderId){return true;}public void closeOrder(String orderId){System.out.println(DateTime.now().toString(JodaUtil.HH_MM_SS)+ " 关闭订单");}
}

执行请求,看一下结果:

在订单创建30秒后,检测到延时队列中有任务任务,调用检测超时方法检测到订单没有完成后,自动关闭订单。

除了上面这种延时队列的方式外,Redisson还提供了另一种方式,也能优雅的关闭订单,方法很简单,就是通过对将要过期的key值的监听。

创建一个类继承KeyExpirationEventMessageListener,重写其中的onMessage方法,就能实现对过期key的监听,一旦有缓存过期,就会调用其中的onMessage方法:

@Component
public class RedisExpiredListener extends KeyExpirationEventMessageListener {public static final String UNPAID_PREFIX="unpaidOrder:";@AutowiredOrderService orderService;public RedisExpiredListener(RedisMessageListenerContainer listenerContainer) {super(listenerContainer);}@Overridepublic void onMessage(Message message, byte[] pattern) {String expiredKey = message.toString();if (expiredKey.startsWith(UNPAID_PREFIX)){System.out.println(DateTime.now().toString(JodaUtil.HH_MM_SS)+" " +expiredKey+"已过期");orderService.closeOrder(expiredKey);}}
}

因为可能会有很多key的过期事件,因此需要对订单过期的key加上一个前缀,用来判断过期的key是不是属于订单事件,如果是的话那么进行关闭订单操作。

再在写一个测试接口,用于创建订单和接收支付成功的回调结果:

@RestController
@RequestMapping("order")
public class TestController {@AutowiredRedisTemplate redisTemplate;@GetMapping("create")public String setTemp(String id){String orderId= RedisExpiredListener.UNPAID_PREFIX+id;System.out.println(DateTime.now().toString(JodaUtil.HH_MM_SS)+" 创建订单:"+orderId);redisTemplate.opsForValue().set(orderId,orderId,30, TimeUnit.SECONDS);return id;}@GetMapping("fallback")public void successFallback(String id){String orderId= RedisExpiredListener.UNPAID_PREFIX+id;redisTemplate.delete(orderId);}
}

在订单支付成功后,一般我们会收到第三方的一个支付成功的异步回调通知。如果支付完成后收到了这个回调,那么我们主动删除缓存的未支付订单,那么也就不会监听到这个订单的orderId的过期失效事件。

但是这种方式有一个弊端,就是只能监听到过期缓存的key,不能获取到对应的value。而通过延时队列的方式,可以通过为RBlockingQueue添加泛型的方式,保存更多订单的信息,例如直接将对象存进队列中:

RBlockingQueue<OrderDTO> blockingFairQueue = redissonClient.getBlockingQueue("orderQueue");
RDelayedQueue<OrderDTO> delayedQueue = redissonClient.getDelayedQueue(blockingFairQueue);

这样的话我们再从延时队列中获取的时候,能够拿到更多我们需要的属性。综合以上两种方式,监听过期更为简单,但存在的一定的局限性,如果我们只需要对订单进行判断的话那么功能也能够满足我们的需求,如果需要在过期时获取更多的订单属性,那么使用延时队列的方式则更为合适。究竟选择哪种,就要看大家的业务场景了。

使用Redisson优雅关闭订单相关推荐

  1. Socket编程中的强制关闭与优雅关闭及相关socket选项

    以下描述主要是针对windows平台下的TCP socket而言. 首先需要区分一下关闭socket和关闭TCP连接的区别,关闭TCP连接是指TCP协议层的东西,就是两个TCP端之间交换了一些协议包( ...

  2. 如何优雅关闭 Spring Boot 应用

    点击蓝色"程序猿DD"关注我 回复"资源"获取独家整理的学习资料! 前言 随着线上应用逐步采用 SpringBoot 构建,SpringBoot应用实例越来多, ...

  3. 白话Elasticsearch71-ES生产集群部署之各个节点以daemon模式运行以及优雅关闭

    文章目录 概述 官方指导 启动 ES 优雅的关闭 ES 概述 继续跟中华石杉老师学习ES,第71篇 课程地址: https://www.roncoo.com/view/55 官方指导 启动ES htt ...

  4. php微信支付分取消订单,微信支付PHP开发教程五关闭订单

    重要:本文最后更新于2019-06-01 08:22:14,某些文章具有时效性,若有错误或已失效,请在下方留言或联系代码狗. 前面我们已经完成了微信支付的正常下单,并且能够正常收款.如果用户在支付过程 ...

  5. Spring Schedule关闭订单

    写一个最简单的版本,因为我们是一个Schedule,不需要返回值,closeOrderTaskV1,第一个版本,那如果其实我们不是TOMCAT集群的话,我们这一个方法就OK了,但是我们是TOMCAT集 ...

  6. 一起学并发编程 - 优雅关闭

    Java中原来在Thread中提供了stop()方法来终止线程,但这个方法是不安全的,所以一般不建议使用.文本将介绍两种可以优雅的终止线程的方式... <!-- more --> 第一种 ...

  7. Spring Boot使用@Async实现异步调用:ThreadPoolTaskScheduler线程池的优雅关闭

    上周发了一篇关于Spring Boot中使用@Async来实现异步任务和线程池控制的文章:<Spring Boot使用@Async实现异步调用:自定义线程池>.由于最近身边也发现了不少异步 ...

  8. GRPC: 如何优雅关闭进程(graceful shutdown)

    简介: 本文将介绍优雅关闭 gRPC 微服务.在进程收到关闭信号时,我们需要关闭后台运行的逻辑,比如,MySQL 连接等等. 介绍 本文将介绍优雅关闭 gRPC 微服务. 什么是优雅关闭? 在进程收到 ...

  9. springboot 优雅关闭_Springboot 优雅停止服务的几种方法

    在使用Springboot的时候,都要涉及到服务的停止和启动,当我们停止服务的时候,很多时候大家都是kill -9 直接把程序进程杀掉,这样程序不会执行优雅的关闭.而且一些没有执行完的程序就会直接退出 ...

最新文章

  1. 特征选择--文本分类: 信息增益
  2. ORACLE基础学习-RMAN应用-控制文件恢复
  3. java小编程---生成不重复的随机数列
  4. iview的走马灯嵌套在模态框中,宽度为0的解决方案
  5. Android中SlidingDrawer开发报错You need to use a Theme.AppCompat theme (or descendant) with this activity.
  6. [渝粤教育] 西南科技大学 施工组织 在线考试复习资料
  7. 支付宝:提现免费再延3年,不设上限!
  8. 浏览器兼容性问题和解决方案
  9. 计算机走进画图世界课件,windowsxp走进画图世界教案
  10. flink 一次job卡deploying故障解决
  11. Q4财报再次显示,百度在2B市场的竞争中已占据领先地位
  12. SpringBoot从入门到精通教程(三十)- 支付宝企业支付集成(五分钟集成)
  13. jdon的设计模式详细解读
  14. 2023年会议教学庭审录像机产品分析
  15. prometheus+alertmanager+webhook实现自定义监控报警系统
  16. 程序员不一定要进大厂,但是算法一定要学
  17. 2017计算机组装,2017电脑组装配置
  18. ChatGPT 大智近妖,从宇宙人生到手搓光刻机,从哄女朋友到写年终总结我们聊得非常开心,反而让人越来越忧心...
  19. matlab 二重傅里叶积分,傅里叶积分、傅里叶变换的matlab实现.doc
  20. osgearth 仿真平台之卫星仿真(实现STK卫星仿真基本功能)

热门文章

  1. 了解下RDF 容器元素
  2. unix oracle控制台,Linux平台下启动oracle11gEM控制台
  3. mysql 查询优化 非索引_mysql 查询优化和索引使用心得
  4. 置换 ---- 2020-2021 ICPC NERC 的 K. King‘s Task[置换类型思维题]
  5. Luogu P4336 [SHOI2016]黑暗前的幻想乡(容斥,矩阵树定理,子集反演)
  6. (每日一题)CF1139D Steps to One 2021年天梯赛 L3-3 可怜的简单题(期望,莫比乌斯反演,杜教筛)
  7. AcWing 2983. 玩具 / POJ 2318.toys(计算几何基础、二分、判断点和直线的位置关系)
  8. 哈尔滨商业大学计算机与信息工程学院地址,计算机与信息工程学院
  9. java session缓存_Java服务端采用Session的缓存oauth2.0授权用户信息
  10. poj3692(二分图最大独立集)