- 在某些程序中,有一些特殊的功能需要用到定时执行,如果熟悉Linux的朋友肯定会说这不是容易吗,直接来个计划任务crontab不久实现了吗?这的确是可以实现,但必须是提前知道具体的执行时间,然后才能写到计划任务里去。比如凌晨两点,上午七点,或者每天上午六点三十分等等。
然而有时候,这个时间我们无法预知,而执行时间是程序动态生成的。然后在动态生成的时间后执行某些程序片段,这里就不能用Linux的crontab计划任务了,因为每次执行的时间都是动态生成,而计划任务需要知道一个定死了的时间。既然无法使用计划任务,那么就只能从程序本身寻找实现方法。
在PHP中,有一个sleep函数,大概意思是程序执行遇到sleep函数时暂停N秒后继续往下执行。如sleep(10)意思就是程序自上往下执行,遇到sleep(10)语句后暂停十秒,然后继续往下执行。函数括号内的参数是一个数值,代表暂停时间值,单位秒。请看下面一段代码
<?php
/**
* sleep函数的用法
* 琼台博客
*/
// 输出当前程序时间戳
echo time(); // out:1338088780
echo '<br />';
// 暂停十秒
sleep(10);
// 输出时间戳
echo time(); // out:1338088790
以上程序执行结果是
复制代码 代码如下:
1338088780
1338088790
- 可能某些童鞋做实例的时候会说怎么我的程序执行出错,提示超时。出现这个问题不要慌,这是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
- 由于服务器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;
}
- 三、众多要执行的定时文件:
这个是真正要执行的代码:包括采集,数据整理与分析等,文件路径写到配置文件的value中去。同一时间执行的文件,记得一维数组模式。
123

被折叠的 条评论
为什么被折叠?



