服务器之家:专注于服务器技术及软件下载分享
分类导航

Mysql|Sql Server|Oracle|Redis|MongoDB|PostgreSQL|Sqlite|DB2|mariadb|Access|数据库技术|

服务器之家 - 数据库 - Redis - 使用Redisson优雅关闭订单

使用Redisson优雅关闭订单

2021-04-21 01:48码农参上Dr Hydra Redis

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

使用Redisson优雅关闭订单

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

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

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

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

  1. <dependency> 
  2.     <groupId>org.redisson</groupId> 
  3.     <artifactId>redisson-spring-boot-starter</artifactId> 
  4.     <version>3.10.7</version> 
  5. </dependency> 

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

  1. @Component 
  2. public class UnpaidOrderQueue { 
  3.     @Autowired 
  4.     RedissonClient redissonClient; 
  5.  
  6.     public void addUnpaid(String orderId){ 
  7.         RBlockingQueue<String> blockingFairQueue = redissonClient.getBlockingQueue("orderQueue"); 
  8.         RDelayedQueue<String> delayedQueue = redissonClient.getDelayedQueue(blockingFairQueue); 
  9.  
  10.         System.out.println(DateTime.now().toString(JodaUtil.HH_MM_SS)+" 添加任务到延时队列"); 
  11.         delayedQueue.offer(orderId,30, TimeUnit.SECONDS); 
  12.     } 

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

  1. @Component 
  2. public class QueueRunner implements CommandLineRunner { 
  3.     @Autowired 
  4.     private RedissonClient redissonClient; 
  5.  
  6.     @Autowired 
  7.     private OrderService orderService; 
  8.  
  9.     @Override 
  10.     public void run(String... args) throws Exception { 
  11.         new Thread(()->{ 
  12.             RBlockingQueue<String> blockingFairQueue = redissonClient.getBlockingQueue("orderQueue"); 
  13.             RDelayedQueue<String> delayedQueue = redissonClient.getDelayedQueue(blockingFairQueue); 
  14.             delayedQueue.offer(null, 1, TimeUnit.SECONDS); 
  15.             while (true){ 
  16.                 String orderId = null
  17.                 try { 
  18.                     orderId = blockingFairQueue.take(); 
  19.                 } catch (Exception e) { 
  20.                     continue
  21.                 } 
  22.                 if (orderId==null) { 
  23.                     continue
  24.                 } 
  25.                 System.out.println(String.format(DateTime.now().toString(JodaUtil.HH_MM_SS)+" 延时队列收到:"+orderId)); 
  26.                 System.out.println(DateTime.now().toString(JodaUtil.HH_MM_SS)+" 检测订单是否完成支付"); 
  27.                 if (orderService.isTimeOut(orderId)) { 
  28.                     orderService.closeOrder(orderId); 
  29.                 } 
  30.             } 
  31.         }).start(); 
  32.     } 

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

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

  1. @Service 
  2. public class OrderService { 
  3.  
  4.     @Autowired 
  5.     UnpaidOrderQueue unpaidOrderQueue; 
  6.  
  7.     public void createOrder(String order){ 
  8.         System.out.println(DateTime.now().toString(JodaUtil.HH_MM_SS)+" 创建订单:"+order); 
  9.         unpaidOrderQueue.addUnpaid(order); 
  10.     } 
  11.  
  12.     public boolean isTimeOut(String orderId){ 
  13.         return true
  14.     } 
  15.  
  16.     public void closeOrder(String orderId){ 
  17.         System.out.println(DateTime.now().toString(JodaUtil.HH_MM_SS)+ " 关闭订单"); 
  18.     } 

执行请求,看一下结果:

使用Redisson优雅关闭订单

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

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

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

  1. @Component 
  2. public class RedisExpiredListener extends KeyExpirationEventMessageListener { 
  3.     public static final String UNPAID_PREFIX="unpaidOrder:"
  4.  
  5.     @Autowired 
  6.     OrderService orderService; 
  7.  
  8.     public RedisExpiredListener(RedisMessageListenerContainer listenerContainer) { 
  9.         super(listenerContainer); 
  10.     } 
  11.  
  12.     @Override 
  13.     public void onMessage(Message message, byte[] pattern) { 
  14.         String expiredKey = message.toString(); 
  15.         if (expiredKey.startsWith(UNPAID_PREFIX)){ 
  16.             System.out.println(DateTime.now().toString(JodaUtil.HH_MM_SS)+" " +expiredKey+"已过期"); 
  17.             orderService.closeOrder(expiredKey); 
  18.         } 
  19.     } 

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

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

  1. @RestController 
  2. @RequestMapping("order"
  3. public class TestController { 
  4.     @Autowired 
  5.     RedisTemplate redisTemplate; 
  6.  
  7.     @GetMapping("create"
  8.     public String setTemp(String id){ 
  9.         String orderId= RedisExpiredListener.UNPAID_PREFIX+id; 
  10.         System.out.println(DateTime.now().toString(JodaUtil.HH_MM_SS)+" 创建订单:"+orderId); 
  11.         redisTemplate.opsForValue().set(orderId,orderId,30, TimeUnit.SECONDS); 
  12.         return id; 
  13.     } 
  14.  
  15.     @GetMapping("fallback"
  16.     public void successFallback(String id){ 
  17.         String orderId= RedisExpiredListener.UNPAID_PREFIX+id; 
  18.         redisTemplate.delete(orderId); 
  19.     } 

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

使用Redisson优雅关闭订单

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

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

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

原文地址:https://mp.weixin.qq.com/s/AMTEuBId3L1LOnpwkQZ5HA

延伸 · 阅读

精彩推荐
  • Redisredis实现排行榜功能

    redis实现排行榜功能

    排行榜在很多地方都能使用到,redis的zset可以很方便地用来实现排行榜功能,本文就来简单的介绍一下如何使用,具有一定的参考价值,感兴趣的小伙伴们...

    乘月归5022021-08-05
  • RedisRedis的配置、启动、操作和关闭方法

    Redis的配置、启动、操作和关闭方法

    今天小编就为大家分享一篇Redis的配置、启动、操作和关闭方法,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧 ...

    大道化简5312019-11-14
  • RedisRedis如何实现数据库读写分离详解

    Redis如何实现数据库读写分离详解

    Redis的主从架构,能帮助我们实现读多,写少的情况,下面这篇文章主要给大家介绍了关于Redis如何实现数据库读写分离的相关资料,文中通过示例代码介绍...

    罗兵漂流记6092019-11-11
  • Redis详解Redis复制原理

    详解Redis复制原理

    与大多数db一样,Redis也提供了复制机制,以满足故障恢复和负载均衡等需求。复制也是Redis高可用的基础,哨兵和集群都是建立在复制基础上实现高可用的...

    李留广10222021-08-09
  • Redisredis中如何使用lua脚本让你的灵活性提高5个逼格详解

    redis中如何使用lua脚本让你的灵活性提高5个逼格详解

    这篇文章主要给大家介绍了关于redis中如何使用lua脚本让你的灵活性提高5个逼格的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具...

    一线码农5812019-11-18
  • RedisRedis 事务知识点相关总结

    Redis 事务知识点相关总结

    这篇文章主要介绍了Redis 事务相关总结,帮助大家更好的理解和学习使用Redis,感兴趣的朋友可以了解下...

    AsiaYe8232021-07-28
  • Redisredis 交集、并集、差集的具体使用

    redis 交集、并集、差集的具体使用

    这篇文章主要介绍了redis 交集、并集、差集的具体使用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友...

    xiaojin21cen10152021-07-27
  • RedisRedis全量复制与部分复制示例详解

    Redis全量复制与部分复制示例详解

    这篇文章主要给大家介绍了关于Redis全量复制与部分复制的相关资料,文中通过示例代码介绍的非常详细,对大家学习或者使用Redis爬虫具有一定的参考学习...

    豆子先生5052019-11-27