当前位置:首页 > 科技  > 软件

数十万定时任务,如何高效触发定时和超时

来源: 责编: 时间:2023-11-07 09:14:10 350观看
导读项目产品中,大家都会有"定时任务"和"定时超时"的需求,初始阶段,我们基本都是用少数的一些timer,即使是任务量越来越大的时候,我们就难免维护着大量的timer,或者进行了大量低效的扫描。定时任务使用场景:当订单一直处于未支付

项目产品中,大家都会有"定时任务"和"定时超时"的需求,初始阶段,我们基本都是用少数的一些timer,即使是任务量越来越大的时候,我们就难免维护着大量的timer,或者进行了大量低效的扫描。C0P28资讯网——每日最新资讯28at.com

C0P28资讯网——每日最新资讯28at.com

定时任务使用场景:当订单一直处于未支付状态时,如何及时的关闭订单(已经使用)C0P28资讯网——每日最新资讯28at.com

如何定期检查处于退款状态的订单是否已经退款成功(后期重构使用)C0P28资讯网——每日最新资讯28at.com

C0P28资讯网——每日最新资讯28at.com

设计方案:C0P28资讯网——每日最新资讯28at.com

  • 整个Redis当做消息池,以KV形式存储消息
  • 使用ZSET做优先队列,按照Score维持优先级
  • 使用LIST结构,以先进先出的方式消费
  • ZSET和LIST存储消息地址(对应消息池的每个KEY)
  • 使用定时器维护路由
  • 根据TTL规则实现消息延迟

C0P28资讯网——每日最新资讯28at.com

咱们公司现阶段就是使用的这套方法:C0P28资讯网——每日最新资讯28at.com

1.新增一个job,会job_pool中插入一条数据,记录了业务方消费方。也会在bucket插入一条记录,记录执行的时间戳C0P28资讯网——每日最新资讯28at.com

2.搬运线程会去bucket中查找哪些执行时间戳的RunTimeMillis比现在的时间小,将这些记录全部删除;同时会解析出每个任务的Topic是什么,然后将这些任务PUSH到TOPIC对应的列表queue中C0P28资讯网——每日最新资讯28at.com

3每个topic的list都会有一个监听线程去批量获取list中的待消费数据,获取到的数据全部扔给这个topic的消费线程池C0P28资讯网——每日最新资讯28at.com

4.消费线程池执行会去job_pool查找数据结构,返回给回调结构,执行回调方法。C0P28资讯网——每日最新资讯28at.com

C0P28资讯网——每日最新资讯28at.com

图片图片C0P28资讯网——每日最新资讯28at.com

待优化的内容:C0P28资讯网——每日最新资讯28at.com

  1. 目前只有一个Queue队列存放消息,当需要消费的消息大量堆积后,会影响消息通知的时效。改进的办法是,开启多个Queue,进行消息路由,再开启多个消费线程进行消费,提供吞吐量
  2. 消息没有进行持久化,存在风险,后续会将消息持久化到MongoDB中

一般来说还有什么其他方法实现这类需求呢?C0P28资讯网——每日最新资讯28at.com

“轮询扫描法”C0P28资讯网——每日最新资讯28at.com

1.用一个Map<uid, last_packet_time>来记录每一个uid最近一次请求时间last_packet_timeC0P28资讯网——每日最新资讯28at.com

2.当某个用户uid有请求包来到,实时更新这个MapC0P28资讯网——每日最新资讯28at.com

3.启动一个timer,当Map中不为空时,轮询扫描这个Map,看每个uid的last_packet_time是否超过30s,如果超过则进行超时处理C0P28资讯网——每日最新资讯28at.com

C0P28资讯网——每日最新资讯28at.com

“多timer触发法”C0P28资讯网——每日最新资讯28at.com

1.用一个Map<uid, last_packet_time>来记录每一个uid最近一次请求时间last_packet_timeC0P28资讯网——每日最新资讯28at.com

2.当某个用户uid有请求包来到,实时更新这个Map,并同时对这个uid请求包启动一个timer,30s之后触发C0P28资讯网——每日最新资讯28at.com

3.每个uid请求包对应的timer触发后,看Map中,查看这个uid的last_packet_time是否超过30s,如果超过则进行超时处理C0P28资讯网——每日最新资讯28at.com

方案一:只启动一个timer,但需要轮询,效率较低C0P28资讯网——每日最新资讯28at.com

方案二:不需要轮询,但每个请求包要启动一个timer,比较耗资源C0P28资讯网——每日最新资讯28at.com

C0P28资讯网——每日最新资讯28at.com

ZSet(有序集合)数据结构来实现C0P28资讯网——每日最新资讯28at.com

  1. 创建ZSet:首先,你需要创建一个ZSet数据结构,其中每个订单将作为一个成员,其分数将表示订单的创建时间戳。你可以使用Redis等支持ZSet的数据库来实现。
  2. 添加订单:每当用户创建新订单时,将订单添加到ZSet中,其中成员是订单ID,分数是订单的创建时间戳。
  3. 定时检查订单:定期(例如,每分钟)执行一个程序或定时任务来检查ZSet中的订单。你可以使用程序来查询ZSet,找到创建时间超过一定时间阈值的订单,表示它们长时间未支付。
  4. 取消订单:对于那些长时间未支付的订单,可以将其从ZSet中删除,并执行取消订单的操作(例如,将订单状态设置为"已取消")。

C0P28资讯网——每日最新资讯28at.com

import redis.clients.jedis.Jedis;import redis.clients.jedis.Tuple;import java.util.Set;public class OrderCancellationSystem {    public static void main(String[] args) {        Jedis jedis = new Jedis("localhost"); // 连接到本地Redis服务器        // 模拟添加订单        addOrder(jedis, "Order1");        addOrder(jedis, "Order2");        // 定时任务,每分钟检查订单并自动取消        while (true) {            cancelLongUnpaidOrders(jedis);            try {                Thread.sleep(60000); // 等待一分钟再次检查            } catch (InterruptedException e) {                e.printStackTrace();            }        }    }    public static void addOrder(Jedis jedis, String orderId) {        long currentTime = System.currentTimeMillis();        jedis.zadd("orders", currentTime, orderId);    }    public static void cancelOrder(String orderId) {        // 执行取消订单操作,例如更新订单状态        System.out.println("Cancelling order: " + orderId);    }    public static void cancelLongUnpaidOrders(Jedis jedis) {        long expirationTime = System.currentTimeMillis() - 3600 * 1000; // 60分钟前的时间戳        Set<Tuple> longUnpaidOrders = jedis.zrangeByScoreWithScores("orders", "-inf", String.valueOf(expirationTime));        for (Tuple order : longUnpaidOrders) {            String orderId = order.getElement();            cancelOrder(orderId);            jedis.zrem("orders", orderId); // 从ZSet中删除已取消的订单        }    }}

C0P28资讯网——每日最新资讯28at.com

本文链接:http://www.28at.com/showinfo-26-17398-0.html数十万定时任务,如何高效触发定时和超时

声明:本网页内容旨在传播知识,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。邮件:2376512515@qq.com

上一篇: 了解 Go 中原子操作的重要性与使用方法

下一篇: Istio Envoy 配置解读,看这篇就够了

标签:
  • 热门焦点
  • 直屏旗舰来了 iQOO 12和K70 Pro同台竞技

    旗舰机基本上使用的都是双曲面屏幕,这就让很多喜欢直屏的爱好者在苦等一款直屏旗舰,这次,你们等到了。据博主数码闲聊站带来的最新爆料称,Redmi下代旗舰K70 Pro和iQOO 12两款手
  • Rust中的高吞吐量流处理

    作者 | Noz编译 | 王瑞平本篇文章主要介绍了Rust中流处理的概念、方法和优化。作者不仅介绍了流处理的基本概念以及Rust中常用的流处理库,还使用这些库实现了一个流处理程序
  • 服务存储设计模式:Cache-Aside模式

    Cache-Aside模式一种常用的缓存方式,通常是把数据从主存储加载到KV缓存中,加速后续的访问。在存在重复度的场景,Cache-Aside可以提升服务性能,降低底层存储的压力,缺点是缓存和底
  • 从 Pulsar Client 的原理到它的监控面板

    背景前段时间业务团队偶尔会碰到一些 Pulsar 使用的问题,比如消息阻塞不消费了、生产者消息发送缓慢等各种问题。虽然我们有个监控页面可以根据 topic 维度查看他的发送状态,
  • 企业采用CRM系统的11个好处

    客户关系管理(CRM)软件可以为企业提供很多的好处,从客户保留到提高生产力。  CRM软件用于企业收集客户互动,以改善客户体验和满意度。  CRM软件市场规模如今超过580
  • 得物效率前端微应用推进过程与思考

    一、背景效率工程随着业务的发展,组织规模的扩大,越来越多的企业开始意识到协作效率对于企业团队的重要性,甚至是决定其在某个行业竞争中突围的关键,是企业长久生存的根本。得物
  • 一个注解实现接口幂等,这样才优雅!

    场景码猿慢病云管理系统中其实高并发的场景不是很多,没有必要每个接口都去考虑并发高的场景,比如添加住院患者的这个接口,具体的业务代码就不贴了,业务伪代码如下:图片上述代码有
  • 小米汽车电池信息疑似曝光:容量101kWh,支持800V高压快充

    7月14日消息,今日一名博主在社交媒体发布了一张疑似小米汽车电池信息的照片,显示该电池包正是宁德时代麒麟电池,容量为101kWh,电压为726.7V,可以预测小
  • 四年持续更迭坚持探索行业无人之境,HarmonyOS 4带来五大升级多项创新

    除了华为每年新发布的旗舰手机系列,上亿花粉更加期待鸿蒙系统每次的跨版本大更新。8月4日,HarmonyOS 4于HDC 2023正式发布,这也是该系统历经四年的再
Top