Laravel怎么执行定时任务_Laravel调度器Task Scheduling配置【实战】

2次阅读

定时任务未执行的主因是服务器crontab未正确配置Laravel调度器,需添加 * cd /path && php artisan schedule:run >> /dev/null 2>&1,并确保php CLI与FPM版本一致、闭包环境兼容、cron表达式精准、跨服务器时启用redis缓存锁。

Laravel怎么执行定时任务_Laravel调度器Task Scheduling配置【实战】

定时任务没跑起来?先确认 crontab 是否真正调用了 laravel 调度器

绝大多数“定时任务不执行”的问题,根源不在 Laravel 代码,而在于服务器的 crontab 没配对。Laravel 的调度器本身不是守护进程,它依赖系统级的 cron 每分钟拉起一次 php artisan schedule:run,再由 Laravel 内部判断哪些任务该执行。

必须在服务器上运行以下命令添加基础 cron 条目:

* * * * * cd /your/project/path && php artisan schedule:run >> /dev/null 2>&1
  • 别漏掉 cd /your/project/path —— 缺少工作目录会导致配置加载失败、日志写入异常
  • >>/dev/null 2>&1 是静默输出,但调试阶段建议先改成 > /var/log/laravel-schedule.log 2>&1 查看真实错误
  • crontab -e 编辑,保存后执行 sudo systemctl restart cron(或 service cron reload)确保生效

schedule:run 执行了但任务没触发?检查闭包任务的 PHP 版本与环境隔离

Laravel 调度器在执行闭包任务(即直接写在 app/console/Kernel.phpschedule 方法里的 ->call(function () { ... }))时,会通过 eval() 或序列化方式运行,这对运行环境敏感。

  • 如果部署在 PHP-FPM + CLI 版本不一致的环境(比如 FPM 用 PHP 8.2,CLI 用 PHP 8.1),闭包可能因语法不兼容报错,且错误不会打到日志里 —— 建议统一 CLI 和 Web 的 PHP 版本
  • 闭包内不能访问 $this,也不能依赖未 use 的变量;若需访问容器服务,改用 ->command('inspire') 或自定义 Artisan 命令
  • 本地测试可用 php artisan schedule:run --verbose 强制触发并打印当前匹配的任务列表

任务执行频率写错?everyFiveMinutes()cron("*/5 * * * *") 行为不同

Laravel 提供两类时间控制:链式方法(如 hourly()dailyAt('9:00'))和原生 cron() 字符串。它们底层都转为 cron 表达式,但精度和语义有差异。

  • everyFiveMinutes() 等链式方法会在每次 schedule:run 被调用时做“是否到达间隔起点”的判断,实际执行窗口可能漂移 ±60 秒
  • cron("*/5 * * * *") 交给系统 cron 控制,更准时,但要求你完全理解 cron 语法(例如 "0 */2 * * *" 表示每两小时整点执行,不是“每隔两小时”)
  • 避免混用:不要对同一任务既写 ->hourly() 又加 ->withoutOverlapping(),后者依赖数据库锁,而前者本身已按小时粒度调度

多台服务器部署时,withoutOverlapping() 失效?默认文件锁不跨机器

withoutOverlapping() 默认使用 file 驱动,在单机有效;但如果你用负载均衡或 K8s 部署多个 Laravel 实例,文件锁无法共享,任务仍可能并发执行。

  • 必须显式切换为 cachedatabase 驱动:->withoutOverlapping()->onOneServer() 不够,要配合 SCHEDULER_CACHE_DRIVER=redis 或设置 'cache' => ['store' => 'redis']
  • onOneServer() 是补充机制:它让所有实例协商选出一个“主节点”来执行任务,依赖缓存的原子操作(如 Redis 的 SETNX),所以 Redis 连接必须稳定且低延迟
  • 数据库驱动需要运行 php artisan schedule:install 创建 scheduled_tasks 表(仅 Laravel 10+ 自动支持)

跨服务器场景下,锁机制不是开箱即用的,得亲手验证缓存键是否存在、TTL 是否合理、网络分区时的行为——这些细节往往在压测或故障时才暴露。

text=ZqhQzanResources