spring boot项目中处理Schedule定时任务
项目中,因为使用了第三方支付(支付宝和微信支付),支付完毕后,第三方支付平台一般会采用异步回调通知的方式,通知商户支付结果,然后商户根据通知内容,变更商户项目支付订单的状态。一般来说,为了防止商户项目自身因为一些特殊原因,比如正好当时网络状态不稳定,商户回调接口无法访问,或者商户回调接口出现异常。第三方支付平台,一般会发送多次请求来尽量确保通知到商户系统。
但是,总会有各种各样的情况,导致,第三方平台所有的通知次数通知完毕后,商户系统依然没有正确处理掉该笔订单状态。(一般第三方支付平台,会:1分钟,10分钟,1小时,6小时,12小时,24小时通知商户支付接口,当然这个时间只是举例而已,实际的通知时间会和这个不太一样)。当如果按这个频次通知完毕后,第三方支付平台就不再进行通知了。
所以,一旦第三方支付平台,将通知次数发送完毕后,平台依然没有处理好自己项目的订单状态,那么订单状态很可能是,客户实际已经完成支付,但是自己项目这边依然显示:“等待支付”这样的状态。那么为了解决这类问题,一般,我们会采用定时任务的方式,比如1个小时,我们把订单支付已经超过1小时,但是还没超过24小时的订单去第三方平台主动的查询一次订单状态。避免因为自己原因,导致的订单状态不一致的情况。这类设置已经超过1小时,未超过24小时的原因,主要是为了查询效率,因为半小时之内还未支付,我们认为实际中存在这个可能性,所以可以等到1个小时,在查状态,而不超过24小时的原因是,一旦超过24小时,我们可以认为订单实际上已经超时失败了,如果查这个订单,就会导致,每次定时任务都会把之前已经查过的订单,在反复的查询。造成大量不必要的查询。一般我们也会在启一个定时任务,24小时执行一次,把一些订单重置为失败。减少不必要的失败订单的反复查询。
下面进入正题,在spring boot中,如何来处理定时任务。
默认,springboot已经支持了定时任务Schedule模块,所以一般情况已经完全能够满足我们的实际需求,一般来说,没有必要在加入其他类似于:quartz
另外,在这里提一个实际项目中,关于定时任务的架构上的一些考虑:
一般来说,实际项目中,为了提高服务的响应能力,我们一般会通过负载均衡的方式,或者反向代理多个节点的方式来进行。通俗点来说,我们一般会将项目部署多实例,或者说部署多份,每个实例不同的启动端口。但是每个实例的代码其实都是一样的。如果我们将定时任务写在我们的项目中,就会面临一个麻烦,就是比如我们部署了3个实例,三个实例一启动,就会把定时任务都启动,那么在同一个时间点,定时任务会一起执行,也就是会执行3次,这样很可能会导致我们的业务出现错误。
一般来说,我们有几种简单的办法来处理:
1、配置文件中增加自定义配置,通过开关来进行控制:比如增加:schedule=enable , schedule=disable,这样在我们的实际代码中,在进行判断,也就是我们可以通过配置,达到,只有一个实例真正执行定时任务,其他的是实例不执行。但是,这种做法实际是还是定时任务都启动,只是在执行中,我们人工来进行判断,执行于不执行真正的处理逻辑。
2、逻辑分离,就是我们将真正要定时任务处理的逻辑,写成rest服务,或者rpc服务,然后我们可以新建一个单独的定时任务项目,这个项目应该是没有任何的业务代码的,他纯粹只有定时任务功能,几点启动,或者每隔多少时间启动,启动后,通过rest或者rpc的方式,调用真正处理逻辑的服务。同时,我们甚至可以不用新建一个项目,我们通过linux的cron就可以进行。同时,这种方式还有一个好处,比如有些时候,我们的定时任务也会因为某些原因出现问题,没有执行,那么我们就可以通过curl 或者wget等等很多方式,再次定时任务的执行。
3、利用redis锁机制。
所以,个人一般偏向使用第二种方式,达到定时任务和业务处理的分离。
参考文章:https://www.jianshu.com/p/ef18af5a9c1d
【推荐】中国电信天翼云云端翼购节,2核2G云服务器一口价38元/年
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步