Java处理延时任务的常用几种解决方案


Posted in Java/Android onJune 01, 2022

前言

项目中经常会遇到如下的需求:

  • 创建订单30分钟未支付,订单自动取消。
  • 订单支付成功后,1分钟后给用户发送短信,提醒用户评价。

针对延时任务需求,我们可以采用如下的解决方案:

Java处理延时任务的常用几种解决方案

数据库轮询

原理

通过一个线程定时的扫描数据库当天创建的订单,根据订单的创建时间来判断订单是否超时,针对超时订单进行相关的更新操作。
实现技术
采用Spring Boot结合quartz来实现,具体的实现可以参考之前的文章。

优缺点

优点:

此方案比较简单,且quartz也支持集群操作。

缺点:

  • 系统订单数据量比较大,每个几分钟轮询数据库,对服务器和数据库的内存消耗比较大。
  • 存在延迟,即使1分钟扫描一次数据库,也会存在1分钟的延迟。

Java延迟队列

原理

采用JDK自带的DelayQueue来实现,这是一个无界阻塞队列,该队列只有在延迟期满的时候才能从中获取元素,放入DelayQueue中的对象。
实现技术
使用JDK的DelayQueue队列进行相关操作即可。

优缺点

优点:

此方案是基于内存操作所以效率高,任务触发时间延迟低.

缺点:

  • 消息队列的信息都存放在内存中,一旦服务器重启,则数据全部消失
  • 无法进行集群用扩展
  • 由于本机内存有限,一旦订单数据量过大,很容易出现OOM异常。

Reids监听失效key

原理

该方案使用Redis的Keyspace Notifications,利用key失效的提供的回调机制,处理相关的业务实现

实现技术

基于reids的方案,实现MessageListener接口。

实现步骤

修改Redis配置文件
打开redis.conf 文件,搜索 “notify-keyspace-events”找到原本的notify-keyspace-events " ",修改为 “notify-keyspace-events Ex”,至此Redis 就支持Key过期事件的监听。

Java处理延时任务的常用几种解决方案

创建监听类,实现MessageListener接口

@Component
public class RedisKeyExpirationListener implements MessageListener
{
    private static final Logger logger = LoggerFactory.getLogger(RedisKeyExpirationListener.class);
    
    public static final String KEY_PREX = "test::order:queue";
    
    @Override
    public void onMessage(Message message, byte[] pattern)
    {
        try
        {
            String expiredKey = message.toString();

            // 通过key来判断
            if(!expiredKey.contains(KEY_PREX))
            {
                return;
            }
            
            //满足条件处理具体的业务逻辑
        }
        catch (Exception e)
        {
            logger.error("失效事件失败",e);
        }
    }
}

优缺点

优点:

基于Redis实现简单

缺点:

  • 客户端断开后重连会导致所有事件丢失。
  • 高并发场景下,存在大量的失效key场景会导出失效时间存在延迟。
  • 此方案针对业务量较少且可靠性要求不高的场景使用。

RocketMq延迟消息

实现原理

基于RocketMQ设置消息的等级,发送延迟消息,RocketMQ延时消息会暂存在名为SCHEDULE_TOPIC_XXXX的Topic中,并根据delayTimeLevel存入特定的queue,queueId = delayTimeLevel – 1,即一个queue只存相同延迟的消息,保证具有相同发送延迟的消息能够顺序消费。broker会调度地消费SCHEDULE_TOPIC_XXXX,将消息写入真实的topic。
其具体步骤如下:

  • 修改消息Topic名称和队列信息
  • 转发消息到延迟主题SCHEDULE_TOPIC_XXXX的CosumeQueue中
  • 延迟服务消费SCHEDULE_TOPIC_XXXX消息
  • 将信息重新存储到CommitLog中
  • 将消息投递到目标Topic中
  • 消费者消费目标topic中的数据。
/**
    * 发送延迟消息
    * @param topic
    * @param msg
    */
   public void sendDelayMessage(String topic,Object msg)
   {
      Message msgMessage =new Message();
      //设置消息等级
      msgMessage.setDelayTimeLevel(2);
     rocketMQTemplate.convertAndSend(topic, msg);
   }

注意:RocketMQ延时消息的延迟时长不支持随意时长的延迟,是通过特定的延迟等级来指定的。默认支持18个等级的延迟消息,延时等级定义在RocketMQ服务端的MessageStoreConfig类中的如下变量中:

Java处理延时任务的常用几种解决方案

例如指定的延时等级为2,则表示延迟时长为5s,即延迟等级是从1开始计数的。

优缺点

优点:

支持高并发场景消息处理.

缺点:

  • 引入额外的消息队列,增加项目的维护和复杂度。
  • 支持固定时长的消息延迟,针对任意时长的消息延迟需要进行扩展。

总结

本文讲解了针对延时任务的处理的几种方案和相关的优缺点,针对不同的业务场景,选择合适的解决方案。更多相关Java 延时任务内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!


Tags in this post...

Java/Android 相关文章推荐
ConstraintValidator类如何实现自定义注解校验前端传参
Jun 18 Java/Android
java中重写父类方法加不加@Override详解
Jun 21 Java/Android
Java图书管理系统,课程设计必用(源码+文档)
Jun 30 Java/Android
Java并发编程必备之Future机制
Jun 30 Java/Android
详细了解MVC+proxy
Jul 09 Java/Android
SpringBoot中HttpSessionListener的简单使用方式
Mar 17 Java/Android
Java基于Dijkstra算法实现校园导游程序
Mar 17 Java/Android
Java实战之课程信息管理系统的实现
Apr 01 Java/Android
Spring Boot 底层原理基础深度解析
Apr 03 Java/Android
Java虚拟机内存结构及编码实战分享
Apr 07 Java/Android
Spring中的@Transactional的工作原理
Jun 05 Java/Android
java.util.NoSuchElementException原因及两种解决方法
Jun 28 Java/Android
Java实现添加条码或二维码到Word文档
Jun 01 #Java/Android
Spring IOC容器Bean的作用域及生命周期实例
May 30 #Java/Android
spring IOC容器的Bean管理XML自动装配过程
May 30 #Java/Android
利用正则表达式匹配浮点型数据
May 30 #Java/Android
JavaScript正则表达式实现注册信息校验功能
May 30 #Java/Android
Java时间工具类Date的常用处理方法
May 25 #Java/Android
Java实现扫雷游戏详细代码讲解
You might like
PHP实现文件安全下载
2006/10/09 PHP
全文搜索和替换
2006/10/09 PHP
PHP strtr() 函数使用说明
2008/11/21 PHP
PHP开发中常用的三个表单验证函数使用小结
2010/03/03 PHP
PHP生成迅雷、快车、旋风等软件的下载链接代码实例
2014/05/12 PHP
简单谈谈php延迟静态绑定
2016/01/26 PHP
Yii框架防止sql注入,xss攻击与csrf攻击的方法
2016/10/18 PHP
thinkPHP中_initialize方法实例分析
2016/12/05 PHP
Iframe thickbox2.0使用的方法
2009/03/05 Javascript
在javascript中关于节点内容加强
2013/04/11 Javascript
ie 7/8不支持trim的属性的解决方案
2014/05/23 Javascript
javascript关于运动的各种问题经典总结
2015/04/27 Javascript
详解JavaScript中的表单验证
2015/06/16 Javascript
JavaScript如何自定义trim方法
2015/07/28 Javascript
JS基于HTML5的canvas标签实现炫目的色相球动画效果实例
2016/08/24 Javascript
jQuery实现CheckBox全选、全不选功能
2017/01/11 Javascript
JavaScript实现鼠标滚轮控制页面图片切换功能示例
2017/10/14 Javascript
小程序scroll-view安卓机隐藏横向滚动条的实现详解
2019/05/16 Javascript
Vue实现一种简单的无限循环滚动动画的示例
2021/01/10 Vue.js
[06:06]2018DOTA2亚洲邀请赛主赛事第四日战况回顾 全明星赛欢乐上演
2018/04/07 DOTA
Python性能优化的20条建议
2014/10/25 Python
Python随机生成均匀分布在单位圆内的点代码示例
2017/11/13 Python
python判断完全平方数的方法
2018/11/13 Python
python basemap 画出经纬度并标定的实例
2019/07/09 Python
pytorch掉坑记录:model.eval的作用说明
2020/06/23 Python
分享unittest单元测试框架中几种常用的用例加载方法
2020/12/02 Python
详解Python+Selenium+ChromeDriver的配置和问题解决
2021/01/19 Python
One.com挪威:北欧成长最快的网络托管公司
2016/11/19 全球购物
naturalizer加拿大官网:美国娜然女鞋
2017/04/04 全球购物
英国街头品牌:Bee Inspired Clothing
2018/02/12 全球购物
美国购买当代和现代家具网站:MODTEMPO
2018/07/20 全球购物
sleep()方法和wait()方法的区别是什么
2012/11/17 面试题
毕业生自我鉴定
2013/11/05 职场文书
工商管理专业毕业生求职信
2014/05/26 职场文书
2015年公司保安年终工作总结
2015/05/14 职场文书
Python基于百度API识别并提取图片中文字
2021/06/27 Python