今天遇到的场景如标题:

支付订单超时自动处理 (待支付订单的有效期三分钟)。

想到的解决方法

1. 定时器

2.redis 【Redis 事件通知(keyspace & keyevent notification)】

3.rabbitmq 延时队列

主要给大家分别讲下 方案2 和 方案3

方案2 :

找到机器上redis配置文件conf/redis.conf,新增一行 notify-keyspace-events Ex

配置文件修改完成后重新启动redis服务。

(如果配置文件修改后未生效,进入redis-cli命令行界面,使用非持久化配置 CONFIG set notify-keyspace-events Ex)

代码如下所示: 例子可以直接跑通。 直接在 TopicMessageListener onMessage 方法下 打断点 即可看到到期redis 值 进入断点中 。

https://github.com/KingestCode/redisnotify.git

方案三:RabbitMQ 延迟消息实现

千万记住,一定选好版本号,由于我使用的是RabbitMQ 3.7.4,因此对应的rabbitmq_delayed_message_exchange插件也必须选择3.7.x的。

下载完插件后,将其放置到RabbitMQ安装目录下的plugins目录下,并使用如下命令启动这个插件:

./rabbitmq-plugins enable rabbitmq_delayed_message_exchange

集成RabbitMQ

org.springframework.boot

spring-boot-starter-amqp

在 application.yml加入 rabbitmq 的配置

spring.rabbitmq.host=127.0.0.1

server:

port: 8081

redis:

hostname: localhost

port: 6379

spring:

rabbitmq:

host: 127.0.0.1

port: 5672

username: guest

password: guest

virtual-host: /

publisher-confirms: true

publisher-returns: true

template:

mandatory: true

connection-timeout: 15000

RabbitMqConfig

@Configuration

@ConfigurationProperties(prefix = "spring.rabbitmq")

public class RabbitMqConfig {

private String host;

private int port;

private String userName;

private String password;

@Bean

public ConnectionFactory connectionFactory() {

CachingConnectionFactory cachingConnectionFactory = new CachingConnectionFactory(host,port);

cachingConnectionFactory.setUsername(userName);

cachingConnectionFactory.setPassword(password);

cachingConnectionFactory.setVirtualHost("/");

cachingConnectionFactory.setPublisherConfirms(true);

return cachingConnectionFactory;

}

@Bean

public RabbitTemplate rabbitTemplate() {

RabbitTemplate rabbitTemplate = new RabbitTemplate(connectionFactory());

return rabbitTemplate;

}

public String getHost() {

return host;

}

public void setHost(String host) {

this.host = host;

}

public int getPort() {

return port;

}

public void setPort(int port) {

this.port = port;

}

public String getUserName() {

return userName;

}

public void setUserName(String userName) {

this.userName = userName;

}

public String getPassword() {

return password;

}

public void setPassword(String password) {

this.password = password;

}

}

QueueConfig

@Configuration

public class QueueConfig {

@Bean

public CustomExchange delayExchange() {

Map args = new HashMap<>();

args.put("x-delayed-type", "direct");

return new CustomExchange("test_exchange", "x-delayed-message",true, false,args);

}

@Bean

public Queue queue() {

Queue queue = new Queue("test_queue_1", true);

return queue;

}

@Bean

public Binding binding() {

return BindingBuilder.bind(queue()).to(delayExchange()).with("test_queue_1").noargs();

}

}

消息发送:MessageServiceImpl

@Service

public class MessageServiceImpl {

@Autowired

private RabbitTemplate rabbitTemplate;

public void sendMsg(String queueName,String msg) {

SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");

System.out.println("消息发送时间:"+sdf.format(new Date()));

rabbitTemplate.convertAndSend("test_exchange", queueName, msg, new MessagePostProcessor() {

@Override

public Message postProcessMessage(Message message) throws AmqpException {

message.getMessageProperties().setHeader("x-delay",3000);

return message;

}

});

}

}

消息接受:MessageReceiver

@Component

public class MessageReceiver {

@RabbitListener(queues = "test_queue_1")

public void receive(String msg) {

SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");

System.out.println("消息接收时间:"+sdf.format(new Date()));

System.out.println("接收到的消息:"+msg);

}

}

测试类: RabbitmqApplicationTests

@RunWith(SpringRunner.class)

@SpringBootTest(classes = UserApplication.class)

public class RabbitmqApplicationTests {

@Autowired

private MessageServiceImpl messageService;

@Test

public void send() {

messageService.sendMsg("test_queue_1","hello i am delay msg");

}

}

运行完后,可以看到如下信息:

消息发送时间:2018-05-03 12:44:53

3秒钟后,Spring Boot控制台会输出:

消息接收时间:2018-05-03 12:44:56

接收到的消息:hello i am delay msg

最后讲下 定时任务以及redis 和 rabbitmq 之间的区别:

其实用定时任务,确实有点问题,原本业务系统希望10分钟后,如果订单未支付,就马上取消订单,并释放商品库存。但是一旦数据量大的话,就会加长获取未支付订单数据的时间,部分订单就做不到10分钟后取消了,可能是15分钟,20分钟之类的。这样的话,库存就无法及时得到释放,也就会影响成单数。而利用延迟消息,则理论上是可以做到按照设定的时间,进行订单取消操作的。

目前网上关于使用RabbitMQ实现延迟消息的文章,大多都是讲如何利用RabbitMQ的死信队列来实现,实现方案看起来都很繁琐复杂,并且还是使用原始的RabbitMQ Client API来实现的,更加显得啰嗦。

Spring Boot 已经对RabbitMQ Client API进行了包装,使用起来简洁很多,下面详细介绍一下如何利用rabbitmq_delayed_message_exchange 插件和Spring Boot来实现延迟消息。

redis 和 rabbitmq 的方式会保证数据的准确性

如果需要rabbitmq 相关代码 请留言 。。。

订单超时自动取消 java_springboot 支付订单超时自动处理相关推荐

  1. Redis key过期事件监听实现 - 30分钟自动取消未支付订单

    目录 一.前言 二.实现方案分析 三.Redis key过期事件方案实现步骤 3.1 Redis 安装步骤详见 3.2 修改 Redis 配置 3.3 在获取支付链接视图中设置key过期事件 3.4 ...

  2. 订单超时自动取消3种方案——我们用这种!

    大家好,大家对电商购物应该都比较熟悉了,我们应该注意到,在下单之后,通常会有一个倒计时,如果超过支付时间,订单就会被自动取消. 下单 今天,我们来聊聊订单超时未支付自动取消的几种方案. 1.定时任务 ...

  3. delayQueue实现订单超时自动取消

    目录 说明 实现 1.编写Delayed实现类 2.编写DelayQueue业务类 3.编写订单业务逻辑 总结说明 说明 商城系统的订单模块都应该有:订单未支付超时后自动取消订单的操作.我们在开发过程 ...

  4. Java 实现订单未支付超时自动取消

    在电商上购买商品后,如果在下单而又没有支付的情况下,一般提示30分钟完成支付,否则订单自动.比如在京东下单为完成支付: 超过24小时,就会自动取消订单,下面使用 Java 定时器实现超时取消订单功能. ...

  5. TP6订单待支付超时自动取消

    1:安装redis 2:安装think-queue composer require topthink/think-queue 3:设置配置文件,使用redis进行存储,文件位置config/queu ...

  6. 使用DelayQueue模拟订单超时自动取消

    1.创建能在DelayQueue中存放的Order对象 package com.example.javastudy.delay_queue;import java.time.Duration; imp ...

  7. 订单过期 自动取消实现方案

    在电商.支付等领域,往往会有这样的场景,用户下单后放弃支付了,那这笔订单会在指定的时间段后进行关闭操作,细心的你一定发现了像某宝.某东都有这样的逻辑,而且时间很准确,误差在1s内:那他们是怎么实现的呢 ...

  8. springboot 模拟秒杀 分布式锁 以及 延时取消未支付订单

    简易描述秒杀系统的几个主要特点,分布式情况下使用锁,订单超时未支付使用mq的延时队列取消 maven依赖 <dependency><groupId>org.springfram ...

  9. 【TWS API 问题3】盈透证券的TWS API使用过程中如何实现一个订单成交之后自动取消其他的订单?

    有读者付费咨询下面一个问题: 请问在陆续所下的n个定单中之一被触发执行后,⚠️在保留该订单子订单不被取消的前提下⚠️,如何快速取消其他全部订单呢?是否有比按照订单ID逐一取消所有订单更快捷的方式呢? ...

  10. Python线程超时自动终止 | Python利用ThreadPoolExecutor实现对多线程的超时自动终止 | Python3实现单线程超时自动强制停止

    文章目录 1. 按 2. 实测代码 1. 按 Python多线程适用于IO密集型的应用场景,与进程不同的是多线程对计算机资源的占用较少.对于Python自带的模块,threading未实现线程池,co ...

最新文章

  1. android如何查看分区信息,android如何查看分区信息
  2. 指定python 版本安装相关插件
  3. python导入txt为dataframe-python读取文本中数据并转化为DataFrame的实例
  4. 2021宁夏英语高考成绩查询,2021宁夏高考成绩官方查询时间及入口
  5. jzoj3501-消息传递【换根法,树形dp】
  6. [Leetcode][第32题][JAVA][最长有效括号][动态规划][栈][正向逆向结合]
  7. Java 集合List、Set、HashMap操作一(Array转List、Set排序、HashMap遍历、Set遍历、List遍历、HashMap大小长度、List打乱顺序)
  8. Leetcode每日一题:904.fruit-into-baskets(水果成篮)
  9. 开启和关闭oracle数据库中的审计功能
  10. Qt信号和槽的连接--发射信号而槽未响应
  11. ARTS打卡第14周
  12. SkinSharp函数文档
  13. VB写的随机点名器代码
  14. 两英中学2021高考成绩查询,2021年广东高中排名,高中高考成绩排名一览表
  15. python是什么意思中文、好学吗-零基础学python难吗?好学吗?
  16. 图片格式网页在线一键转换源码
  17. 博世XC事业部李胤:自动驾驶降温不意外,但这条路肯定会走下去
  18. oracle数据库ORA-报错大全
  19. 计算机网络延展-桥接器(网桥)
  20. linux给串口赋予权限

热门文章

  1. 用Python写爬虫代码,记录斗鱼主播的热度变化情况,并绘制热度变化图。
  2. 深入理解多线程(四)—— Moniter的实现原理
  3. android+色调+半透明,91桌面Android3.0绽放 智能美化双管齐下
  4. 短网址生成+域名检测+短网址还原+域名防红四合一前端源码
  5. html 缩小时字体错位,微信字体设置导致页面错位的解决方法
  6. [CF1538E] Funny Substrings (模拟)
  7. 4 个方法养成大神级 “反内耗“ 体质
  8. 基于单幅图像Patch Map的稳健除雾(PMS-Net: Robust Haze Removal Based on Patch Map for Single Images_CVPR_2019)
  9. Ubuntu 如何安装 微信?实测
  10. R语言小代码(聚类练习)