Laravel 技巧之 定时任务

Stella981
• 阅读 822

定时任务 Scheduled Tasks 是 Laravel 提供的组件之一,稍微上点规模的项目应该都会用到,比如开发微信应用时通过定时任务去刷新access token,比如每天定时发推送提现用户要记得签到。对于定时任务的基本用法,官网文档已经描述得很详细了,这里不再多说。

本文主要是介绍定时任务在实际应用中的两个小技巧:

1. 多个任务并行执行

先简单介绍一下 Laravel 定时任务组件的基本原理:

当cli初始化完毕之后,系统会调用 App\Console\Kernel::schedule 方法,也就是我们定义定时任务列表的地方,这个方法里每调用一次 $schedule->command() 就会生成一个 Illuminate\Console\Scheduling\Event 对象并保存在 $schedule->events 数组里。当执行 php artisan scheduled:run 时,系统会遍历 $schedule->events,把当前时间需要执行的任务放在一个集合中,最后依次 串行执行 这些任务。

这样做在大多数情况下是没有问题的,但有一些特殊的情况,比如在每个月的第一天要给100W个用户发送邮件,同一批次的定时任务必须等到这些邮件全部发送完毕之后才会被执行,假如这些任务里有对执行时间十分敏感的任务,比每5分钟一次的数据快照,就会导致那个时间点数据的缺失。

这种情况下如果定时任务能够并行执行,就不会有这样的问题。Laravel 实际上提供了解决方案,但很奇怪文档里面并没有提到,就是 runInBackground 方法,在定义定时任务时 $schedule->command('foo:bar')->everyMinutes()->runInBackground(); 就可以了。

2. 负载均衡

随着业务逻辑的增多,定时任务也会越来越多,定时任务服务器的负载也会越来越高,甚至导致任务执行缓慢,然而我们却只能在一台服务器上设置定时任务,如果在多台服务器上同时配置了定时任务,还会导致定时任务的重复执行。这个时候我们希望能够像队列那样,将定时任务分散到多台服务器上。

截止 v5.4.15,Laravel 还没有提供内置方案来解决这个问题,但只需要简单的改造就可以实现我们需要的效果。首先我们把将每个定时任务里 handle 方法提取出来创建一个新的Job并继承 ShouldQueue,然后在定时任务的 handle 里直接 dispatch 对应的Job即可,这样原本的业务逻辑就会被队列处理掉,当系统有多台服务器在处理队列时,也就实现了我们需要的负载均衡。

但是这样毕竟还是麻烦,每个定时任务都要创建一个Command和一个Job,太费劲,于是我提交了一个 Proposal ,目前已经实现并且merge入5.4分支,相信下个版本大家就能用上了。用法也很简单,只需要创建一个继承 ShouldQueue的Job,然后在App\Console\Kernel::schedule 方法里定义

$schedule->job(new FooBarJob())->everyMinutes();

就可以了

点赞
收藏
评论区
推荐文章
Wesley13 Wesley13
3年前
java目前可以通过以下几种方式进行定时任务
1、单机部署模式Timer:jdk中自带的一个定时调度类,可以简单的实现按某一频度进行任务执行。提供的功能比较单一,无法实现复杂的调度任务。ScheduledExecutorService:也是jdk自带的一个基于线程池设计的定时任务类。其每个调度任务都会分配到线程池中的一个线程执行,所以其任务是并发执行的,互不影响。
Wesley13 Wesley13
3年前
java 定时任务之Timer
java定时任务之TimerTimer是什么?Timer位于java.util包中。官方API的描述是用来控制任务执行的,每个任务可以执行一次,也可以执行多次。在实际应用中,我们可以用它来控制某个任务在特定的时间执行,或者按照某个固定频率或者时间间隔执行。Timer怎么用?它提供了三类方
Stella981 Stella981
3年前
SpringBoot学习笔记:计划任务
SpringBoot学习笔记:计划任务计划任务在企业的实践生产中,可能需要使用一些定时任务,如月末、季末和年末需要统计各种各样的报表,每周自动备份数据等。在Spring中使用定时任务1、加入@EnableScheduling注解,以启用定时任务机制@EnableS
Stella981 Stella981
3年前
Spring Boot 与 Kotlin 定时任务(Scheduling Tasks)
在编写SpringBoot应用中会遇到这样的场景,比如:需要定时地发送一些短信、邮件之类的操作,也可能会定时地检查和监控一些标志、参数等。创建定时任务在SpringBoot中编写定时任务是非常简单的事,下面通过实例介绍如何在SpringBoot中创建定时任务,实现每过5秒输出一下当前时间。在SpringBoot的主类中加入
Stella981 Stella981
3年前
Linux的定时任务
任务计划的条件:1.在未来的某个时间点执行一次某个任务(atbatch)2.周期性的执行某个任务(cron)at在指定时间执行任务_用法_at\选项参数\\时间\_选项参数_\l      查看作业\c      显示即将执行任务的细节\d      使用任务id号
Wesley13 Wesley13
3年前
PHP之定时任务(绝对实用)
在工作中经常会用到定时任务,除了在js中有定时器这个玩意可以用于前端页面的定时任务;那么后端PHP如何设置定时任务呢?一.如下是javascript中使用setTimeout和setInterval设置定时任务:1setTimeout(function(){2console.log('timeout');
Stella981 Stella981
3年前
Quartz.NET总结(四)Quartz 远程调度
转载自:http://www.cnblogs.com/zhangweizhong/p/5552558.html 前面篇已经介绍了Quartz.NET的配置,使用和Cron表达式表达式的写法。基本上后台的定时任务的定时执行已经完成,并能正确的按照执行计划,执行相关的job。  然后,如果任务需要更新,停止某个任务呢?总不能上服务器去改相关
Wesley13 Wesley13
3年前
4、定时任务关闭超时未支付的订单
//1.在主启动类上加上支持定时任务的注解@EnableSchedulingpublicclassApplication{//2.编写定时任务@ComponentpublicclassOrderJob{@Autowiredp
Easter79 Easter79
3年前
SpringBoot学习笔记:计划任务
SpringBoot学习笔记:计划任务计划任务在企业的实践生产中,可能需要使用一些定时任务,如月末、季末和年末需要统计各种各样的报表,每周自动备份数据等。在Spring中使用定时任务1、加入@EnableScheduling注解,以启用定时任务机制@EnableS
定时任务原理方案综述 | 京东云技术团队
本文主要介绍目前存在的定时任务处理解决方案。业务系统中存在众多的任务需要定时或定期执行,并且针对不同的系统架构也需要提供不同的解决方案。京东内部也提供了众多定时任务中间件来支持,总结当前各种定时任务原理,从定时任务基础原理、单机定时任务(单线程、多线程)、分布式定时任务介绍目前主流的定时任务的基本原理组成、优缺点等。希望能帮助读者深入理解定时任务具体的算法和实现方案。