php定时任务

来源:互联网 发布:留香久的香水知乎 编辑:程序博客网 时间:2024/06/06 09:53
  1. 在某些程序中,有一些特殊的功能需要用到定时执行,如果熟悉Linux的朋友肯定会说这不是容易吗,直接来个计划任务crontab不久实现了吗?这的确是可以实现,但必须是提前知道具体的执行时间,然后才能写到计划任务里去。比如凌晨两点,上午七点,或者每天上午六点三十分等等。

然而有时候,这个时间我们无法预知,而执行时间是程序动态生成的。然后在动态生成的时间后执行某些程序片段,这里就不能用Linux的crontab计划任务了,因为每次执行的时间都是动态生成,而计划任务需要知道一个定死了的时间。既然无法使用计划任务,那么就只能从程序本身寻找实现方法。

在PHP中,有一个sleep函数,大概意思是程序执行遇到sleep函数时暂停N秒后继续往下执行。如sleep(10)意思就是程序自上往下执行,遇到sleep(10)语句后暂停十秒,然后继续往下执行。函数括号内的参数是一个数值,代表暂停时间值,单位秒。请看下面一段代码

<?php/** * sleep函数的用法 * 琼台博客 */// 输出当前程序时间戳echo time();  // out:1338088780echo '<br />';// 暂停十秒sleep(10);// 输出时间戳echo time(); // out:1338088790以上程序执行结果是复制代码 代码如下:13380887801338088790
  1. 可能某些童鞋做实例的时候会说怎么我的程序执行出错,提示超时。出现这个问题不要慌,这是PHP默认页面执行时间导致的,在PHP中默认执行页面时间是三十秒钟,这对于一般程序够用了。但如果你要做类似定时执行功能就必须再头部声明设置下执行时间set_time_limit(0)。0是代表不限时,单位是秒。最后整体贴出代码:
<?php/** 1. sleep函数定时执行动态生成时间段的代码 2. 琼台博客 */// 设置页面执行时间,否则会有超时错误提示set_time_limit(0);// 目前时间echo date('Y-m-d H:i:s'); // out:2012-05-27 14:58:00// 动态生成时间 范围在今天下午六点到晚上零点前的任意时间$datetime = date('Y-m-d').' '.rand('18,23').':'.rand('0,59').':'.rand('0,59');  // 2012-05-27 19:20:00// 算得时间戳$a = strtotime($datetime);// 算得时间差$reduce = $a-time();// sleep等待sleep($reduce);// 执行到时间后执行的代码块echo date('Y-m-d H:i:s');  // out:2012-05-27 19:20:00
  1. 由于服务器crontab只能精确到分钟,因此程序的起点也是分钟。

该功能一共包括三个部分:

一、配置文件:

配置文件是用来返回要执行的定时任务文件,注意一下*的使用就行了,有两个模式,就是

Y-m-d H:i :年 月 日 时 分
N H:i :星期(1 - 7|周一 - 周日) 时 分

配置文件croning.php如下:

/** * 任务管理器配置文件 *  * Y-m-d H:i    :年 月 日 时 分 * N H:i      :星期(1 - 7|周一 - 周日) 时 分 *  * 2013-12-25 19:49 : 固定时间,只执行一次 * *-12-25 20:00  : 每年的某月某日 某小时某分 * 2013-12-25 *:49 : 某天的每个小时的49分都执行一次 * *-*-* 20:00   : 每天晚上8点0分执行 * *-*-* *:*    :每分钟都在执行 * 2 20:01     :每周二的20:01时间都执行一次 *  * * 号表示当前位置的任何时间。以此类推.... *  * 格式: * array( *   key=>value, * ); *  * 说明: * key是定义的执行时间,value是执行的文件,可以是数组或者字符串,当同一时间有多个任务执行时,为了避免key的覆盖请用一维数组模式。 *  */return array(  '2013-12-25 19:49'=>'123.php',  '2013-12-* 18:00'=>'456.php',  '1 08:00'=>'6546.php',  '*-12-25 19:49'=>array('444.php','456.php'));

二、服务器cronjob主要执行的php文件:

该php文件主要处理与分析哪些文件是当时可以执行的。以及写入执行记录文件。

<?php/** 1. cron任务统一执行的文件,没有超时 */header('Content-Type:text/html; charset=utf-8');set_time_limit(0);define('APP_ROOT', dirname(__FILE__));define('AHA_ROOT', dirname(APP_ROOT));define('CORE_ROOT', AHA_ROOT . '/__core');define('DATA_ROOT', AHA_ROOT . '/data');define('MODEL_ROOT', APP_ROOT . '/model');define('ONING_ROOT', APP_ROOT . '/oning'); //定时执行文件目录require CORE_ROOT . '/Common.php';require CORE_ROOT . '/AHA.php'; //载入框架核心文件spl_autoload_register(array('Common', 'loadClassFile'));AHA::initConfig(include APP_ROOT . '/_config/inc.php'); //载入配置文件//不存在执行的配置文件时if (!file_exists(APP_ROOT . '/_config/croning.php')) {  exit('cron failed,please check the cron config!');}$__all = include APP_ROOT . '/_config/croning.php';//数据不合法时if (!$__all || !is_array($__all)) {  exit('cron failed,please check the cron config!');}$__echo = true; //是否输出到屏幕$__time_star = microtime(true);$__now = time();Common::fileLog(DATA_ROOT . '/log/cron_index.log', '执行cron开始******************************' . date('Y-m-d H:i:s', $__now) . '******************************', $__echo);$__onFile = array();if ($__all) {  foreach ($__all as $__key => $__value) {    if (strpos($__key, '-') === false) {//每周的处理      preg_match('@^([\d\*]+) ([\d\*]+):([\d\*]+)$@U', $__key, $match);    } else {//正常的处理      preg_match('@^([\d\*]+)\-([\d\*]+)\-([\d\*]+) ([\d\*]+):([\d\*]+)$@U', $__key, $match);    }    if ($match) {      array_shift($match);      if (__getPreg($match, $__now)) {//是否是要执行的文件        $__onFile = array_merge($__onFile, is_array($__value) ? $__value : array($__value));      }    }  }}if ($__onFile) {  $__onFile = array_unique($__onFile);  foreach ($__onFile as $__value) {    if (file_exists(ONING_ROOT . '/' . $__value)) {      $__time_star2 = microtime(true);      Common::fileLog(DATA_ROOT . '/log/cron_index.log', $__value . ' 执行开始----------' . date('Y-m-d H:i:s') . '-----------', $__echo);      include ONING_ROOT . '/' . $__value;      Common::fileLog(DATA_ROOT . '/log/cron_index.log', $__value . ' 执行结束(花费时间:' . ((microtime(true) - $__time_star2) * 1000) . 'ms)-------------', $__echo);    }  }}Common::fileLog(DATA_ROOT . '/log/cron_index.log', '执行cron结束(一共执行时间:' . ((microtime(true) - $__time_star) * 1000) . 'ms)*************' . date('Y-m-d H:i:s') . '*****************' . "\n\n", $__echo);/** 2. 处理正则结果并返回该文件是否是当时要执行 3. @param array $match   正则结果,数组 4. @param integer $__now  当时时间戳 5. @return bool */function __getPreg($match, $__now) {  $back = false;  list($__Y, $__m, $__d, $__N, $__H, $__i) = explode('-', date('Y-m-d-N-H-i', $__now));  $argc = count($match);  if ($argc === 3) {    $argc = $match[0] === '*' ? $__N : $match[0];    $argc.=' ';    $argc.=$match[1] === '*' ? $__H : $match[1];    $argc.=':';    $argc.=$match[2] === '*' ? $__i : $match[2];    $back = date('N H:i', $__now) === date($argc, $__now) ? true : false;  } elseif ($argc === 5) {    $argc = $match[0] === '*' ? $__Y : $match[0];    $argc.='-';    $argc.=$match[1] === '*' ? $__m : $match[1];    $argc.='-';    $argc.=$match[2] === '*' ? $__d : $match[2];    $argc.=' ';    $argc.=$match[3] === '*' ? $__H : $match[3];    $argc.=':';    $argc.=$match[4] === '*' ? $__i : $match[4];    $back = date('Y-m-d H:i', $__now) === date($argc, $__now) ? true : false;  }  return $back;}
  1. 三、众多要执行的定时文件:

这个是真正要执行的代码:包括采集,数据整理与分析等,文件路径写到配置文件的value中去。同一时间执行的文件,记得一维数组模式。

0 0
原创粉丝点击