定制化开发订单超时未支付自动取消8种实现方案

定时轮询

定制化开发数据库定时轮询方式,定制化开发实现思路比较简单。定制化开发启动一个定时任务,定制化开发每隔一定时间扫描订单表,定制化开发查询到超时订单就取消。

优点:实现简单。
缺点:定制化开发轮询时间间隔不好确定,定制化开发占用服务器资源,定制化开发影响数据库性能。

惰性取消

定制化开发当查询订单信息时,定制化开发先判断该订单是否超时,定制化开发如果超时就先取消。

优点:实现简单。
缺点:定制化开发影响查询之外的业务(如:统计、库存),定制化开发影响查询效率。

JDK延迟

JDK延时队列DelayQueue是一个无界阻塞队列,该队列只有在延迟期满的时候才能从中获取元素。

简单实现代码demo如下,实际生产过程中会有专门的线程负责消息的入队与消费。

  1. import java.util.concurrent.Delayed;
  2. import java.util.concurrent.TimeUnit;
  3. /**
  4. * @author 向振华
  5. * @date 2022/08/16 15:55
  6. */
  7. public class OrderDelayed implements Delayed {
  8. /**
  9. * 延迟时间
  10. */
  11. private final Long time;
  12. /**
  13. * 订单编号
  14. */
  15. public String orderNo;
  16. public OrderDelayed(String orderNo, long time, TimeUnit unit) {
  17. this.orderNo = orderNo;
  18. this.time = System.currentTimeMillis() + (time > 0 ? unit.toMillis(time) : 0);
  19. }
  20. @Override
  21. public long getDelay(TimeUnit unit) {
  22. return time - System.currentTimeMillis();
  23. }
  24. @Override
  25. public int compareTo(Delayed o) {
  26. OrderDelayed orderDelayed = (OrderDelayed) o;
  27. long diff = this.time - orderDelayed.time;
  28. if (diff <= 0) {
  29. return -1;
  30. } else {
  31. return 1;
  32. }
  33. }
  34. }
  1. import java.util.concurrent.DelayQueue;
  2. import java.util.concurrent.TimeUnit;
  3. /**
  4. * @author 向振华
  5. * @date 2022/08/16 16:02
  6. */
  7. public class Test {
  8. public static void main(String[] args) {
  9. DelayQueue<OrderDelayed> delayQueue = new DelayQueue<>();
  10. delayQueue.put(new OrderDelayed("220101001", 8, TimeUnit.SECONDS));
  11. delayQueue.put(new OrderDelayed("220101002", 4, TimeUnit.SECONDS));
  12. System.out.println("订单延迟队列开始执行");
  13. while (true) {
  14. // 取队列头部元素是否过期
  15. OrderDelayed task = delayQueue.poll();
  16. if (task != null) {
  17. // 取消订单业务逻辑
  18. System.out.println("订单 ---> " + task.orderNo + " 已过期准备取消");
  19. }
  20. }
  21. }
  22. }

 

优点:效率高,任务触发时间延迟低。
缺点:异常恢复困难,集群扩展麻烦,内存占用。

时间轮

时间轮算法类似于时钟,会按某一个方向按固定频率轮动。可以用Netty的HashedWheelTimer来实现时间轮方法。

  1. <dependency>
  2. <groupId>io.netty</groupId>
  3. <artifactId>netty-all</artifactId>
  4. <version>4.1.78.Final</version>
  5. </dependency>
  1. import io.netty.util.HashedWheelTimer;
  2. import io.netty.util.Timeout;
  3. import io.netty.util.Timer;
  4. import io.netty.util.TimerTask;
  5. import java.util.concurrent.TimeUnit;
  6. /**
  7. * @author 向振华
  8. * @date 2022/08/16 16:02
  9. */
  10. public class Test {
  11. public static void main(String[] args) {
  12. // 初始化时间轮
  13. Timer timer = new HashedWheelTimer();
  14. // 定时任务
  15. TimerTask task1 = new TimerTask() {
  16. public void run(Timeout timeout) throws Exception {
  17. // 取消订单业务逻辑
  18. System.out.println("订单1已过期准备取消");
  19. }
  20. };
  21. // 注册此定时任务(延迟时间为5秒,也就是说5秒后订单过期)
  22. timer.newTimeout(task1, 5, TimeUnit.SECONDS);
  23. // 定时任务
  24. TimerTask task2 = new TimerTask() {
  25. public void run(Timeout timeout) throws Exception {
  26. // 取消订单业务逻辑
  27. System.out.println("订单2已过期准备取消");
  28. }
  29. };
  30. // 注册此定时任务(延迟时间为3秒,也就是说3秒后订单过期)
  31. timer.newTimeout(task2, 3, TimeUnit.SECONDS);
  32. }
  33. }

优点:效率高,任务触发时间延迟更低。
缺点:异常恢复困难,集群扩展麻烦,内存占用。

Redis过期

Redis的key过期回调事件,也能达到延迟队列的效果。

在redis.conf加入一条配置:

notify-keyspace-events Ex

监听配置

  1. @Configuration
  2. public class RedisListenerConfig {
  3. @Bean
  4. RedisMessageListenerContainer container(RedisConnectionFactory connectionFactory) {
  5. RedisMessageListenerContainer container = new RedisMessageListenerContainer();
  6. container.setConnectionFactory(connectionFactory);
  7. return container;
  8. }
  9. }

Redis过期回调监听方法

  1. @Component
  2. public class RedisKeyExpirationListener extends KeyExpirationEventMessageListener {
  3. public RedisKeyExpirationListener(RedisMessageListenerContainer listenerContainer) {
  4. super(listenerContainer);
  5. }
  6. @Override
  7. public void onMessage(Message message, byte[] pattern) {
  8. // 过期key,可以设置成订单号
  9. String expiredKey = message.toString();
  10. // 取消订单业务逻辑
  11. System.out.println("订单 ---> " + expiredKey + " 已过期准备取消");
  12. }
  13. }

优点:数据不易丢失,集群扩展方便。
缺点:需要额外维护redis。

Redis有序

Redis的数据结构Zset,同样可以实现延迟队列的效果,主要利用它的score属性,redis通过score来为集合中的成员进行从小到大的排序。通过zadd命令向队列delayqueue 中添加元素,并设置score值表示元素过期的时间。

消费端轮询队列delayqueue, 将元素排序后取最小时间与当前时间比对,如小于当前时间代表已经过期移除key。

  1. public void pollOrderQueue() {
  2. while (true) {
  3. Set<Tuple> set = jedis.zrangeWithScores(delayqueue, 0, 0);
  4. String value = ((Tuple) set.toArray()[0]).getElement();
  5. int score = (int) ((Tuple) set.toArray()[0]).getScore();
  6. int nowSecond = System.currentTimeMillis() / 1000);
  7. if (nowSecond >= score) {
  8. jedis.zrem(delayqueue, value);
  9. System.out.println(sdf.format(new Date()) + " removed key:" + value);
  10. }
  11. if (jedis.zcard(delayqueue) <= 0) {
  12. System.out.println(sdf.format(new Date()) + " zset empty ");
  13. return;
  14. }
  15. Thread.sleep(1000);
  16. }
  17. }

优点:数据不易丢失,集群扩展方便。
缺点:可能重复消费同一key。

任务调度

使用任务调度中间件xxl-job、ScheduleX、Elastic-Job等来实现,设置一个调度时间cron,到达订单过期的调度时间时,触发任务执行取消订单业务逻辑。

例如使用xxl-job实现,订单创建时提交一个过期任务到xxl-job服务器,下面时执行器方法:

  1. import com.xxl.job.core.handler.annotation.XxlJob;
  2. import org.springframework.stereotype.Component;
  3. /**
  4. * @author 向振华
  5. * @date 2022/08/16 15:55
  6. */
  7. @Component
  8. public class JobHandler {
  9. @XxlJob("payExpireJobHandler")
  10. public void payExpireJobHandler(String executorParam) {
  11. // 取消订单业务逻辑
  12. System.out.println("订单 ---> " + executorParam + " 已过期准备取消");
  13. }
  14. }

优点:时效性强,支持分布式。
缺点:实现复杂,维护成本高。

使用RocketMQ、RabbitMQ、Kafka的延时消息,消息在发送到消息队列服务端后并不会立马投递,而是根据消息中的属性延迟固定时间后才投递给消费者。

RocketMQ发送延时消息的示例代码如下:

  1. import com.aliyun.openservices.ons.api.Message;
  2. import com.aliyun.openservices.ons.api.ONSFactory;
  3. import com.aliyun.openservices.ons.api.Producer;
  4. import com.aliyun.openservices.ons.api.PropertyKeyConst;
  5. import com.aliyun.openservices.ons.api.SendResult;
  6. import java.util.Properties;
  7. public class Test {
  8. public static void main(String[] args) {
  9. Properties properties = new Properties();
  10. // AccessKey ID阿里云身份验证,在阿里云RAM控制台创建。
  11. properties.put(PropertyKeyConst.AccessKey, "XXX");
  12. // AccessKey Secret阿里云身份验证,在阿里云RAM控制台创建。
  13. properties.put(PropertyKeyConst.SecretKey, "XXX");
  14. // 设置TCP接入域名,进入消息队列RocketMQ版控制台实例详情页面的接入点区域查看。
  15. properties.put(PropertyKeyConst.NAMESRV_ADDR,
  16. "XXX");
  17. Producer producer = ONSFactory.createProducer(properties);
  18. // 在发送消息前,必须调用start方法来启动Producer,只需调用一次即可。
  19. producer.start();
  20. Message msg = new Message(
  21. // 您在消息队列RocketMQ版控制台创建的Topic。
  22. "Topic",
  23. // Message Tag,可理解为Gmail中的标签,对消息进行再归类,方便Consumer指定过滤条件在消息队列RocketMQ版服务器过滤。
  24. "tag",
  25. // Message Body可以是任何二进制形式的数据,消息队列RocketMQ版不做任何干预,需要Producer与Consumer协商好一致的序列化和反序列化方式。
  26. "Hello MQ".getBytes());
  27. // 设置代表消息的业务关键属性,请尽可能全局唯一。
  28. // 以方便您在无法正常收到消息情况下,可通过控制台查询消息并补发。
  29. // 注意:不设置也不会影响消息正常收发。
  30. msg.setKey("ORDERID_100");
  31. try {
  32. // 延时消息,在指定延迟时间(当前时间之后)进行投递。最大可设置延迟40天投递,单位毫秒(ms)。
  33. // 以下示例表示消息在3秒后投递。
  34. long delayTime = System.currentTimeMillis() + 3000;
  35. // 设置消息需要被投递的时间。
  36. msg.setStartDeliverTime(delayTime);
  37. SendResult sendResult = producer.send(msg);
  38. // 同步发送消息,只要不抛异常就是成功。
  39. if (sendResult != null) {
  40. System.out.println(new Date() + " Send mq message success. Topic is:" + msg.getTopic() + " msgId is: " + sendResult.getMessageId());
  41. }
  42. } catch (Exception e) {
  43. // 消息发送失败,需要进行重试处理,可重新发送这条消息或持久化这条数据进行补偿处理。
  44. System.out.println(new Date() + " Send mq message failed. Topic is:" + msg.getTopic());
  45. e.printStackTrace();
  46. }
  47. // 在应用退出前,销毁Producer对象。
  48. // 注意:如果不销毁也没有问题。
  49. producer.shutdown();
  50. }
  51. }

RocketMQ延时消息的订阅与普通消息订阅一致。

优点:高效,好扩展,支持分布式。
缺点:实现复杂,维护成本高。

网站建设定制开发 软件系统开发定制 定制软件开发 软件开发定制 定制app开发 app开发定制 app开发定制公司 电商商城定制开发 定制小程序开发 定制开发小程序 客户管理系统开发定制 定制网站 定制开发 crm开发定制 开发公司 小程序开发定制 定制软件 收款定制开发 企业网站定制开发 定制化开发 android系统定制开发 定制小程序开发费用 定制设计 专注app软件定制开发 软件开发定制定制 知名网站建设定制 软件定制开发供应商 应用系统定制开发 软件系统定制开发 企业管理系统定制开发 系统定制开发