laravel异步监控定时调度器实例详解


Posted in PHP onJune 21, 2019

定时调度器是什么

laravel默认提供了一个命令定时任务的功能,在其他的php框架下面,没有这个定时任务,我们要跑一些异步脚本怎么操作呢,只能依赖我们系统提供的crontab来做,这就导致我们每次发版本新增定时任务都要去服务器更改crontab代码,获取更新这个配置。

执行命令是php artisan schedule:run 来执行,那放在哪里执行呢,没错这个调起还是需要依赖我们crontab来执行,但是只需要配置一次,后续所有定时任务都在我们业务代码进行控制

场景

我们有一个导入数据的定时任务

//每分钟导入库数据
$schedule->command(self::SIGNATURE)->withoutOverlapping()->everyMinute()->runInBackground();

这里导入长时间最好使用runInBackground(),表示异步执行,其实就是在shell脚本的末尾加上 & 符号,在linux上完全依赖系统的方式完成。

这里使用了withoutOverlapping() 表示在同一时刻只能有一个任务执行,主要逻辑使用的是排它锁实现,依赖于我们cache的driver,我这里使用的是redis,后面作为锁的过期直接redis提供的key过期来做。

出现问题

这个任务在正常情况下都是非常完美的,因为同一时刻只有一个再跑,跑完就可以,但是一个场景出现
有一天我们的qa同学刚部署环境后,我们服务端就在默默的导入库了,因为使用withoutOverlapping($expire_at=1440)这个时候在redis就有一个锁产生了,这个默认带参数是锁的过期时间,默认是一天,然后因为我们docker环境需要更改参数然后进行后端server服务的重启,我们重启也是比较暴力,就是直接发送kill的信号,导致所有在里面跑的进程瞬间kill,而这时候我们的redis的锁缺还存在,而且是1440分钟左右,那当我们server再启动后,发现锁一直存在,没办法进行后续的操作了,只能等着。

解决

那我把锁的时间减少行不行,原来1天,我改成30分钟,没问题,开始第一版方案我们也是这样做,官方也是可以这样做的。

后来我们一想,能否做到一个监控程序呢,进程退出后立马监控到过期呢,这样就不用固定一个时间,这当然是所有软件开发者理想状态:要你开你就开,我挂了锁也就去掉了,不论正常与否。

解决方案

laravel异步监控定时调度器实例详解

说明:

  • 这里命令启动时候,获取进程的pid,然后fork子进程,可以将这个pid传递给子进程。
  • 子进程每隔10s进行一个探活,获取父进程的id与传入的pid是否一致,这里普及一个知识点,如果父进程异常退出,这个子进程未退出就会被init进程(pid=1)接管,那么这就是一个孤儿进程。
  • 同时子进程每次探活的时候就会更改redis的锁的过期时间,如果探活时间间隔是10s,那么我们的过期时间设置就是14s,多冗余一点时间。

代码实现

代码实现总是那么苍白无力哈,这里就写一个laravel的扩展来做,好处就是不影响我们主体的任何代码就完成了,我们的laravel可以随意升级。

github地址:github.com/zzh78727258…

composer地址:packagist.org/packages/ze…

总结

整体实现没有使用判断进程是否存在的ps grep等命令,因为我们docker环境不一定支持这些命令,只是用简单的pid与parent_id做对比。

laravel的在命令开始于结束都进行钩子方式,我们在Listener下面进行监听即可

public function subscribe($events)
 {
  $events->listen(
   [
    CommandStarting::class, // 命令开始的时候
   ],
   __CLASS__ . '@handle'
  );
 }

整体代码是基于laravel扩展化的,不会影响laravel的升级操作。

好了,以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对三水点靠木的支持。

PHP 相关文章推荐
定制404错误页面,并发信给管理员的程序
Oct 09 PHP
PHP教程 变量定义
Oct 23 PHP
php计算十二星座的函数代码
Aug 21 PHP
基于php中使用excel的简单介绍
Aug 02 PHP
11个PHPer必须要了解的编程规范
Sep 22 PHP
php遍历目录方法小结
Mar 10 PHP
WordPress中查询文章的循环Loop结构及用法分析
Dec 17 PHP
24条货真价实的PHP代码优化技巧
Jul 28 PHP
PHP标准类(stdclass)用法示例
Sep 28 PHP
PHP获取IP地址所在地信息的实例(使用纯真IP数据库qqwry.dat)
Nov 15 PHP
php从数据库读取数据,并以json格式返回数据的方法
Aug 21 PHP
PHP simplexml_import_dom()函数讲解
Feb 03 PHP
apache集成php7.3.5的详细步骤
Jun 20 #PHP
PHP基础之输出缓冲区基本概念、原理分析
Jun 19 #PHP
PHP进阶学习之Geo的地图定位算法详解
Jun 19 #PHP
PHP进阶学习之依赖注入与Ioc容器详解
Jun 19 #PHP
yii2 在控制器中验证请求参数的使用方法
Jun 19 #PHP
php自定义排序uasort函数示例【二维数组按指定键值排序】
Jun 19 #PHP
windows 2008r2+php5.6.28环境搭建详细过程
Jun 18 #PHP
You might like
thinkPHP5框架设置404、403等http状态页面的方法
2018/06/05 PHP
php设计模式之装饰模式应用案例详解
2019/06/17 PHP
ExtJS 简介 让你知道extjs是什么
2008/12/29 Javascript
基于jquery的横向滚动条(滑动条)
2011/02/24 Javascript
扩展JavaScript功能的正确方法(译文)
2012/04/12 Javascript
js修改input的type属性及浏览器兼容问题探讨与解决
2013/01/23 Javascript
js异步加载的三种解决方案
2013/03/04 Javascript
JavaScript控制按钮可用或不可用的方法
2015/04/03 Javascript
javascript实现鼠标移到Image上方时显示文字效果的方法
2015/08/07 Javascript
JavaScript中关键字 in 的使用方法详解
2016/10/17 Javascript
AngularJS全局警告框实现方法示例
2017/05/18 Javascript
jQuery实现base64前台加密解密功能详解
2017/08/29 jQuery
vue的过滤器filter实例详解
2018/09/17 Javascript
Postman的下载及安装教程详解
2018/10/16 Javascript
python实现在windows服务中新建进程的方法
2015/06/30 Python
分享Python字符串关键点
2015/12/13 Python
Python实现判断给定列表是否有重复元素的方法
2018/04/11 Python
python pandas dataframe 按列或者按行合并的方法
2018/04/12 Python
Python实现Dijkstra算法
2018/10/17 Python
Python Flask框架扩展操作示例
2019/05/03 Python
通过python实现随机交换礼物程序详解
2019/07/10 Python
Python 实现输入任意多个数,并计算其平均值的例子
2019/07/16 Python
python实现超市管理系统(后台管理)
2019/10/25 Python
python可以用哪些数据库
2020/06/22 Python
使用CSS3代码绘制可爱的Hello Kitty猫
2016/08/03 HTML / CSS
一份比较全的PHP面试题
2016/07/29 面试题
护士自荐信
2013/10/25 职场文书
八年级历史教学反思
2014/01/10 职场文书
普罗米修斯教学反思
2014/02/06 职场文书
运动会入场式解说词
2014/02/18 职场文书
绿色学校实施方案
2014/03/31 职场文书
商业计算机应用专业自荐书
2014/06/09 职场文书
青岛导游词
2015/02/12 职场文书
统计工作个人总结
2015/03/03 职场文书
自从在 IDEA 中用了热部署神器 JRebel 之后,开发效率提升了 10(真棒)
2021/06/26 Java/Android
pd.drop_duplicates删除重复行的方法实现
2022/06/16 Python