shell脚本作为保证PHP脚本不挂掉的守护进程实例分享


Posted in PHP onJuly 15, 2013

前几天开始跑一份数据名单,名单需要提供用户名、是否有手机号、是否有邮箱,用户名单我轻易的获取到了,但是,用户名单有2000w之多,并且去检测用户是否有手机号、是否有邮箱必须得通过一个对外开放的安全接口一个一个用户去请求,然后分析返回值才能知道。

下面是我处理的方案:
1、将2000w名单保存到临时数据表
2、用PHP程序每次从该表获取500个用户,检测完后生成SQL update原纪录
3、为了防止PHP程序突然断掉,用shell脚本每隔1分钟检测,PHP挂掉了则重启
我使用shell脚本作为守护进程的原因是,手机与邮箱的检测接口速度慢,不可能在1~2天将2000w用户检测完。

方案详细:
1、临时保存用户名单表users,表结构如下:

CREATE TABLE `users` ( 
  `account` varchar(50) COMMENT '用户名', 
  `has_phone` tinyint(3) unsigned NOT NULL default '0' COMMENT '是否有手机号', 
  `has_email` tinyint(3) unsigned NOT NULL default '0' COMMENT '是否有邮箱', 
  `flag` tinyint(3) unsigned  NOT NULL default '0' COMMENT '标志位', 
  PRIMARY KEY  (`account`), 
  KEY `flag` (`flag`) 
 ) ENGINE=InnoDB  DEFAULT CHARSET=utf8 COMMENT='名单表';

我先将2000多w用户名导入到这个临时表,has_phone与has_email这二个字段默认都是0(没有),标志位flag说明该用户是否已经检测完。
下面是一部分表数据:
9873aaa,0,0,0
adddwwwd876222,0,0,0
testalexlee,0,0,0
codejia.net,0,0,0
haohdouywaa21,0,0,0
 
2、PHP脚本check_users.php
将 用户名单导入到表之后,再写一个简单的PHP脚本,思路是这样的:每次循环从表取flag=0的500个用户,然后请求接口判断用户是否有手机号、邮箱, 生成一条SQL,保存到一个SQLS数组里,等500个用户全部检测完了之后,循环SQLS数组,更新表里这500个名单,并将flag标志位设置为1, 表示已经检测完,下次就不获取了。
由于PHP脚本代码较长,这里分享下简单的代码说明:
<?php 
class Users{ 
    private $data; 
    private $sqls; 
    private $nums;         //判断是否有500用户 
    private $total_nums;   //当前已经检测完的用户数量     //每次取500个用户 
    private function getUsers(){...} 
    //检测这500个用户并生成SQL 
    private function checkUserInfo(){...} 
    //更新这500个用户 
    private function updateUserInfo(){...} 
    //运行 
    public function run(){ 
        $flag = true; 
        while($flag){ 
             if($this->nums != 500){ $flag = false; } 
             if($this->total_nums == 10000){  
                exit(0); //跑完1w个用户就退出,由守护进程启动 
             } 
             $this->getUsers(); 
             $this->checkUserInfo(); 
             $this->updateUserInfo(); 
             sleep(1); //跑完500用户休息1秒,保护用户检测接口      
         } 
    } 
} 
$user = new Users(); 
$user->run(); 
?>

上面是简洁版的PHP脚本,大概意思到了,刚开始的版本是没有$total_nums这个变量,是因为刚开始跑这个脚本的时候,发现只跑完了4w多条脚本就挂球了,后来一看,是因为连接数据库没连上,脚本一直挂在那里。加上这个变量也无法解决这个问题,只是在每次跑完1w个用户之后,PHP脚本退出,再由下面的shell脚本重新启动。

3、shell脚本作为守护进程
我把这个shell脚本加到了crontab里边,每隔1分钟执行一次,这个shell脚本很简单,检测check_users.php是否存在进程id,如果存在,则说明PHP脚本还在运行,shell脚本不做任何操作;如果不存在,则说明PHP脚本已经exit(0)跑完了1w用户退出了,那么shell脚本启动该脚本,进入下一个1w用户名单的检测。
上面我有讲到,如果PHP脚本在连接数据库的时候,无法连接上的时候,PHP会一直挂球在那里,无法退出了。我在shell脚本里加了一个时间检测,当PHP脚本进程存在的时候,计算已经存在了多长时间,如果超过了我预想的时间,则将PHP脚本kill掉,再重启。

开头的举例数据,结果类似如下:
testalexlee,1,0,1
codejia.net,0,0,1
haohdouywaa21,1,1,1
9873aaa,0,1,1
adddwwwd876222,1,0,1

说在最后:以上用户名单数据只是举个栗子,不要太认真,2000w数据,我估计要跑一段时间了,因为检测接口比较慢,接口在接到请求后还要连表,查表,再返回。其实,最好的方法还是直接从接口请求的表拉一份名单出来,再用shell命令处理下很快就有结果了,可是在公司就是这样,有些东西不开放的,你懂的~~~

PHP 相关文章推荐
php5数字型字符串加解密代码
Apr 24 PHP
php smarty 二级分类代码和模版循环例子
Jun 16 PHP
php生成静态文件的多种方法分享
Jul 17 PHP
浅谈web上存漏洞及原理分析、防范方法(文件名检测漏洞)
Jun 29 PHP
WIN8.1下搭建PHP5.6环境
Apr 29 PHP
Yii2隐藏frontend/web和backend/web的方法
Dec 12 PHP
php仿微信红包分配算法的实现方法
May 13 PHP
php单链表实现代码分享
Jul 04 PHP
php微信公众号js-sdk开发应用
Nov 28 PHP
利用Laravel事件系统如何实现登录日志的记录详解
May 20 PHP
对php 判断http还是https,以及获得当前url的方法详解
Jan 15 PHP
tp5(thinkPHP5框架)时间查询操作实例分析
May 29 PHP
PHP 伪静态技术原理以及突破原理实现介绍
Jul 12 #PHP
php的$_FILES的临时储存文件与回收机制实测过程
Jul 12 #PHP
php图像处理函数大全(推荐收藏)
Jul 11 #PHP
PHP 获取远程文件大小的3种解决方法
Jul 11 #PHP
深入Nginx + PHP 缓存详解
Jul 11 #PHP
基于PHP中的常用函数回顾
Jul 11 #PHP
PHP 获取文件权限函数介绍
Jul 11 #PHP
You might like
ThinkPHP CURD方法之order方法详解
2014/06/18 PHP
laravel 5.3中自定义加密服务的方案详解
2017/05/09 PHP
PHP SFTP实现上传下载功能
2017/07/26 PHP
PHP实现通过strace定位故障原因的方法
2018/04/29 PHP
用js实现的抽象CSS圆角效果!!
2007/05/03 Javascript
js 编写规范
2010/03/03 Javascript
js获取对象为null的解决方法
2013/11/21 Javascript
JavaScript 封装一个tab效果源码分享
2015/09/15 Javascript
JS实现的论坛Ajax打分效果完整实例
2015/10/31 Javascript
JS中的数组方法笔记整理
2016/07/26 Javascript
JavaScript实现简单的日历效果
2016/09/25 Javascript
canvas实现粒子时钟效果
2017/02/06 Javascript
bootstrap 设置checkbox部分选中效果
2017/04/20 Javascript
jQuery实现全选、反选和不选功能
2017/08/16 jQuery
详解Angular2学习笔记之Html属性绑定
2018/01/03 Javascript
vue将毫秒数转化为正常日期格式的实例
2018/09/16 Javascript
js校验开始时间和结束时间
2020/05/26 Javascript
[53:18]Spirit vs Liquid Supermajor小组赛A组 BO3 第三场 6.2
2018/06/03 DOTA
python模块之sys模块和序列化模块(实例讲解)
2017/09/13 Python
浅谈python新式类和旧式类区别
2019/04/26 Python
利用Python检测URL状态
2019/07/31 Python
Python实现打印实心和空心菱形
2019/11/23 Python
Pyqt5 关于流式布局和滚动条的综合使用示例代码
2020/03/24 Python
Django ORM 查询表中某列字段值的方法
2020/04/30 Python
Python实现计算图像RGB均值方式
2020/06/04 Python
关于Kotlin中SAM转换的那些事
2020/09/15 Python
简述你对Statement,PreparedStatement,CallableStatement的理解
2013/03/25 面试题
交通法规咨询中心工作职责
2013/11/27 职场文书
《太阳》教学反思
2014/02/21 职场文书
保险专业大学生职业规划书
2014/03/03 职场文书
揭牌仪式主持词
2014/03/19 职场文书
纪检干部现实表现材料
2014/08/21 职场文书
单位工作证明书格式
2014/10/04 职场文书
幼儿园语言教学反思
2016/02/23 职场文书
Python 把两层列表展开平铺成一层(5种实现方式)
2021/04/07 Python
使用numpy实现矩阵的翻转(flip)与旋转
2021/06/03 Python