用户端设置定时任务怎么实现最好?

常见的延时消息、定时任务比较简单,实现方法也多,但是如果定时任务来自用户设置该怎么办?

需求是用户(商家)可以设置自己店铺商品定时上下架、定时修改库存,而且这个定时任务它不是一次性的,可以一次性、按日、周(任意N天)、月(任意N天)重复。

并且一个用户可以设置多条、一万个用户每人设置5条就是5万个定时任务。

怎么实现最优雅且可靠?

:) wink
唐章明
《L02 从零构建论坛系统》
以构建论坛项目 LaraBBS 为线索,展开对 Laravel 框架的全面学习。应用程序架构思路贴近 Laravel 框架的设计哲学。
《G01 Go 实战入门》
从零开始带你一步步开发一个 Go 博客项目,让你在最短的时间内学会使用 Go 进行编码。项目结构很大程度上参考了 Laravel。
最佳答案

这种刚好做过,我们是物联网,场景类似也是用户增加定时任务或者场景联动。要想高效,实现一个基于时间轮定时任务即可。

2周前 评论
讨论数量: 10

可以有用户访问店铺或者商家访问店铺的时候,定时任务的时间过了就处理

2周前 评论
唐章明 (楼主) 2周前
slowlyo

建个任务表, 存:

  • 周期 (一次性/日/周/月...)
  • 任务类型 (上/下架/改库存...)
  • 关联数据 (需要操作的数据, 比如 goods_id ...)
  • 下一次执行的时间 (next_run_at , 记得加索引)
  • 状态 (执行中, 已完成 ...)
  • ...

laravel 的任务调度, 每分钟查一次 next_run_at 小于当前时间的任务, 执行

数据量大的话, 执行任务可以放到队列中处理, 一个生产者和多个消费者这样

大概思路如上

上面说的访问数据的时候处理, 也是一种常见的手段, 在前置中间件查询当前的数据是否存在关联的, 待处理的定时任务, 如果存在则先处理任务, 再处理正常的业务, 可以达到消除定时任务误差的效果~

2周前 评论

RabbitMQ两种方法

  1. 延迟队列+消费者检查逻辑再重新加入延迟队列

    • 延迟任务数量中等可使用 数万数十万不推荐
  2. TTL + DLX 无限循环

    • 停止或调整较麻烦

如果用 Cron+DB的话会有分钟级左右误差

2周前 评论

大量的定制延时任务、定时任务采用“时间轮”方案处理比较合适,不了解的话可以自己问一下chatgpt、deepseek

2周前 评论

这种刚好做过,我们是物联网,场景类似也是用户增加定时任务或者场景联动。要想高效,实现一个基于时间轮定时任务即可。

2周前 评论

大厂一般有toc超时中心去做,我觉得你可以看看阿里的toc超时设计,通过对引入redis-cluster架构,然后使用有序集合数据结构进行处理,不如你有很多任务,那里分片的时候多分点,这样的存储的超时任务就很多

2周前 评论

逻辑都一样,利用 laravel 定时去查数据库处理就行。做过很多类似功能了。二楼思路也已经讲的很明白了

2周前 评论

我们有类似的。功能不过是提示用户的,我是每分钟查询一次,然后把查询到的放到队列里面进行处理

1周前 评论

如果使用laravel的定时任务来实现 感觉你服务的配置会需要很大 或者说用laravel的定时任务不太合理

因为有这么多定时任务 肯定不能串行执行

那么就只能后台运行(backgroud)后台运行就是每一个任务都是一个进程

如果你同时有1000个任务 就是1000个进程

如果是量不是很大 可以用laravel的定时任务 从数据库里面读取动态加载 像这样

protected function schedule(Schedule $schedule): void
    {
        // $schedule->command('inspire')->hourly();

        ScheduleTask::getSortItems()
            ->each(function (ScheduleTask $item) use ($schedule) {
                if ($item->isDisabled() && 'schedule:finish' !== request()->server('argv')[1]) {
                    return;
                }

                $command = "$item->command --is_schedule_run --schedule_task_id=$item->id";
                $event = $schedule->command($command)->cron($item->expression);

                if ($item->isTaskLock()) {
                    $event->withoutOverlapping($item->task_lock_time ?: $event->expiresAt);
                }

                if ($item->isServerLock()) {
                    $event->onOneServer();
                }

                if ($item->isRunInBackground()) {
                    $event->runInBackground();
                }

                if ('' !== $item->timezone) {
                    $event->timezone($item->timezone);
                }

                if ($item->environments) {
                    $event->environments($item->environments);
                }
            });
    }
1周前 评论

讨论应以学习和精进为目的。请勿发布不友善或者负能量的内容,与人为善,比聪明更重要!