既然是定时器,毋庸置疑,当然是定时器…
时间轮算法:
想象一下时钟,有时针、分针、秒针,秒针走60格,分针走一格,分针走60格,时针走1格,时针走24格及1天。这个算法就是模拟时钟的方式来实现定时器的功能。拿时钟举例,我们把时钟分为3格轮子,分别代码时分秒,然后划分对应的格子,把定时任务按照定时时间来放到对应的格子上,然后开始计时,秒针每转动一次则取格子对应的任务。
这里有个问题,秒针只有60个格子,如果我的定时任务设置的时间大于60秒呢?假如我定时是按每分钟,或者每小时怎么办?这也很简单,把时间格式化成分钟/小时级放到分钟/小时轮子上对应的格子里,然后等分针/时针转动的时候按剩余时间把任务下降到下一层等待秒针的到来,然后执行,如果是重复执行的任务,则执行完成后再计算下一次执行时间放回去。
下面是我盗的一张图(盗图可耻,仅供参考)
实现代码:获取代码
<?php
class TimerWheel
{
public $timerManager;
public $arrTask;
public $tvrBits = 8;
public $tvnBits = 6;
public $tvrSize;
public $tvnSize;
public $tvrMask;
public $tvnMask;
public $maxTval = (1 << 32) - 1;
public $fc = ['s', 'i', 'H', 'd', 'm', 'w'];
public $ary = [60, 60, 24, 0, 12, 7];
public $uJiffies = 0; //当前时间
public $lastTime;
public $logFile;
public function __construct()
{
$this->tvrSize = 1 << $this->tvrBits;
$this->tvnSize = 1 << $this->tvnBits;
$this->tvrMask = $this->tvrSize - 1;
$this->tvnMask = $this->tvnSize - 1;
$this->timerManager = range(0, 4);
for ($i = 0; $i < 5; $i++) {
$this->timerManager[$i] = [];
if ($i == 0) {
$size = $this->tvrSize;
} else {
$size = $this->tvnSize;
}
for ($j = 0; $j < $size; $j++) {
$this->timerManager[$i][$j] = [];
}
}
$this->ary[3] = date('t');
$this->lastTime = time();
}
/**
* uExpires
* format a recently exec time, then make a expires time
* receive a frequency like crontab * * * * * *
* @param string $frequency
* @return number
*/
public function uExpires(string $frequency)
{
$fr = explode(" ", $frequency);
$fields = []; //列举多维度的有效时间数组
for ($i = 0; $i < count($fr); $i++) {
if ($fr[$i] == '*') {
$fields[$i] = range(0, $this->ary[$i] - 1);
} elseif (preg_match("/\*\/(\d*)/", $fr[$i], $matches)) {
$fields[$i] = ['f' => $matches[1]];
} elseif (preg_match("/(\d*)-(\d*)/", $fr[$i], $matches)) {
$fields[$i] = range($matches[1], $matches[2]);
} else {
$fields[$i] = explode(",", $fr[$i]);
}
}
//print_r($fields);
/* 查找一个最近的有效时间 */
$incr = 0;
$year = date("Y");
$tk = [0,0,0,0,0,0];//时间指针数组,代表查找到的每一个纬度最近时间
for ($i = 0; $i < count($fr); $i++) {
$d = date($this->fc[$i]);
if (!isset($fields[$i]['f'])) {
// 检查合适的时间位置
for ($j = 0; $j < count($fields[$i]); $j++) {
if (($d <= $fields[$i][$j] && $j == 0) || ($d <= $fields[$i][$j] && $d > $fields[$i][$j-1])) {
if ($incr) {
if ($fields[$i][$j] == $d) {
$j++;
}
//有进位则把前面的指针都置为零
foreach ($tk as $k => &$v) {
if (isset($fields[$k]['f'])) {
$fields[$k][0] = ($d + $fields[$k]['f']) % $this->ary[$k];
} else {
$v = 0;
}
}
}
break;
}
}
//echo $i,' ',$incr,' '.$j." ";
if (!isset($fields[$i][$j])) {
$j = 0;
if ($i == 4) {
$year++;
$incr = 0; //消除进位
//有进位则把前面的指针都置为零
foreach ($tk as $k => &$v) {
if (isset($fields[$k]['f'])) {
$fields[$k][0] = ($d + $fields[$k]['f']) % $this->ary[$k];
} else {
$v = 0;
}
}
} else {
$incr = 1; //产生进位
}
} else {
$incr = 0; //消除进位
}
$tk[$i] = $j;
} else {
if ($incr) {
$incr = 0;
//有进位则把前面的指针都置为零
foreach ($tk as $k => &$v) {
if (isset($fields[$k]['f'])) {
$fields[$k][0] = ($d + $fields[$k]['f']) % $this->ary[$k];
} else {
$v = 0;
}
}
$fields[$i][0] = $d + $fields[$i]['f'] + 1;
} else {
$fields[$i][0] = $d + $fields[$i]['f'];
}
if ($fields[$i][0] >= $this->ary[$i]) {
$incr = 1;
}
$fields[$i][0] %= $this->ary[$i];
}
}
//按照指针数组合成最近执行时间
$datetime = $year."-".$fields[4][$tk[4]]."-".$fields[3][$tk[3]]." ".$fields[2][$tk[2]].":".$fields[1][$tk[1]].":".$fields[0][$tk[0]];
$this->printLog(__FUNCTION__." ".date("Y-m-d H:i:s")." ".$frequency." => ".$datetime);
return (int)((strtotime($datetime) - (time() - $this->uJiffies))); //定时器到期时间
}
/**
* create a task
* @param unknown $frequency
* @param unknown $id
* @return unknown[]|number[]
*/
public function createTask($frequency, $id)
{
$this->printLog("-------------------------begin createTask---------------------------");
if (isset($this->arrTask[$id])) {
$this->arrTask[$id]['fre'] = $frequency;
$this->arrTask[$id]['uExpires'] = $this->uExpires($frequency);
} else {
$task = ['fre' => $frequency, 'id' => $id, 'uExpires' => $this->uExpires($frequency)];
$this->arrTask[$id] = $task;
}
$this->printLog(__FUNCTION__." ".json_encode($this->arrTask[$id]));
$this->addTask($this->arrTask[$id]);
$this->printLog("--------------------------end createTask----------------------------");
return $task;
}
/**
* add task
* @param unknown $task
*/
public function addTask($task)
{
$id = $task['id'];
$uExpires = $task['uExpires']; //距离定时器到期时间
$uDueTime = $uExpires - $this->uJiffies; //触发剩余时间
//$this->printLog(__FUNCTION__." ".$uDueTime." ".json_encode($this->arrTask));
if ($uDueTime < $this->tvrSize) {
$i = $uExpires & $this->tvrMask;
$this->timerManager[0][$i][$id] = &$this->arrTask[$id];
$this->arrTask[$id][1] = 0; //轮的层级
} elseif ($uDueTime < 1 << ($this->tvrBits + $this->tvnBits)) {
$i = ($uExpires >> $this->tvrBits) & $this->tvnMask;
$this->timerManager[1][$i][$id] = &$this->arrTask[$id];
$this->arrTask[$id][1] = 1;
} elseif ($uDueTime < 1 << ($this->tvrBits + $this->tvnBits * 2)) {
$i = ($uExpires >> ($this->tvrBits + $this->tvnBits)) & $this->tvnMask;
$this->timerManager[2][$i][$id] = &$this->arrTask[$id];
$this->arrTask[$id][1] = 2;
} elseif ($uDueTime < 1 << ($this->tvrBits + $this->tvnBits * 3)) {
$i = ($uExpires >> ($this->tvrBits + $this->tvnBits * 2)) & $this->tvnMask;
$this->timerManager[3][$i][$id] = &$this->arrTask[$id];
$this->arrTask[$id][1] = 3;
} elseif ($uDueTime < 1 << ($this->tvrBits + $this->tvnBits * 4)) {
$i = ($uExpires >> ($this->tvrBits + $this->tvnBits * 3)) & $this->tvnMask;
$this->timerManager[4][$i][$id] = &$this->arrTask[$id];
$this->arrTask[$id][1] = 4;
}
$this->arrTask[$id][2] = $i; //轮中的槽
$this->printLog(__FUNCTION__." ".json_encode($this->arrTask[$id])." ".$uDueTime);
//echo __FUNCTION__." ".json_encode($this->arrTask[$id])."\n";
}
public function delTask($id)
{
if (isset($this->arrTask[$id])) {
unset($this->timerManager[$this->arrTask[$id][1]][$this->arrTask[$id][2]][$id]);
unset($this->arrTask[$id]);
}
return 1;
}
public function updateTask($id)
{
if (isset($this->arrTask[$id])) {
unset($this->timerManager[$this->arrTask[$id][1]][$this->arrTask[$id][2]][$id]);
$this->createTask($this->arrTask[$id]['fre'], $this->arrTask[$id]['id']);
}
return 1;
}
/**
* cascade update task
* @param unknown $lv
*/
public function cascadeTask($lv)
{
if ($lv == 1) {
$i = $this->uJiffies >> $this->tvrBits;
} else {
$i = $this->uJiffies >> ($this->tvrBits + $this->tvnBits * ($lv) - 1);
}
if ($i > 0 && 0 == $i % $this->tvnSize) { //判断时间进位
$this->cascadeTask($lv+1);
}
$i %= $this->tvnSize;
$this->printLog("lv=".$lv." i=".$i);
foreach ($this->timerManager[$lv][$i] as $id => $task) {
unset($this->timerManager[$lv][$i][$id]);
$this->addTask($task);
}
}
public function getTask()
{
$taskList = [];
$now = time();
//$now = $this->lastTime+1;
if ($this->lastTime != $now) {
//$this->printLog("++++++++++++++++++++++++++++begin getTask++++++++++++++++++++++++++++");
while ($this->lastTime != $now) { //避免程序执行产生的时间误差导致跳过>1秒
if ($this->uJiffies != 0 && ($this->uJiffies % $this->tvrSize) == 0) {
$this->printLog("jiff=".$this->uJiffies);
$this->cascadeTask(1);
}
$taskList = array_merge($taskList, $this->timerManager[0][$this->uJiffies % $this->tvrMask]);
$this->uJiffies++; //瞬间时间++
$this->lastTime++;
foreach ($taskList as $task) {
$this->printLog(__FUNCTION__." ".json_encode($task)." ".($this->uJiffies-1));
$this->updateTask($task['id']);
}
}
//$this->printLog("+++++++++++++++++++++++++++++end getTask+++++++++++++++++++++++++++++");
}
return $taskList;
}
public function setLogFile($logFile)
{
$this->logFile = $logFile;
}
public function printLog($msg)
{
if (is_array($msg)) {
$msg = json_encode($msg);
}
$msg = date("Y-m-d H:i:s")." ".posix_getpid()." ".$msg."\n";
if ($this->logFile) {
file_put_contents($this->logFile, $msg, FILE_APPEND | LOCK_EX);
} else {
echo $msg;
}
}
}
$timer = new TimerWheel();
$timer->createTask("* 15 12-20 * * *", 333);
$timer->createTask("15 10-15 * * * *", 333);
$timer->createTask("*/10 * * * * *", 222);
$timer->createTask("15 */2 * * * *", 333);
$timer->createTask("25 14 */3 09 * *", 555);
$timer->createTask("25 10 */3 * * *", 555);
$timer->createTask("* * * * * *", 111);
$timer->createTask("10 * * * * *", 222);
$timer->createTask("15 10 * * * *", 333);
$timer->createTask("* 10,20,30 * * * *", 334);
$timer->createTask("20 12 08 * * *", 444);
$timer->createTask("* * 08 * * *", 445);
$timer->createTask("25 14 18 09 * *", 555);
$timer->createTask("* * * 09 * *", 556);
$timer->createTask("30 12 * * 1 *", 666);
$timer->createTask("* 12 * * 1 *", 666);
for ($i = 0; $i < 100; $i++) {
$timer->getTask();
}
class TimerWheel
{
public $timerManager;
public $arrTask;
public $tvrBits = 8;
public $tvnBits = 6;
public $tvrSize;
public $tvnSize;
public $tvrMask;
public $tvnMask;
public $maxTval = (1 << 32) - 1;
public $fc = ['s', 'i', 'H', 'd', 'm', 'w'];
public $ary = [60, 60, 24, 0, 12, 7];
public $uJiffies = 0; //当前时间
public $lastTime;
public $logFile;
public function __construct()
{
$this->tvrSize = 1 << $this->tvrBits;
$this->tvnSize = 1 << $this->tvnBits;
$this->tvrMask = $this->tvrSize - 1;
$this->tvnMask = $this->tvnSize - 1;
$this->timerManager = range(0, 4);
for ($i = 0; $i < 5; $i++) {
$this->timerManager[$i] = [];
if ($i == 0) {
$size = $this->tvrSize;
} else {
$size = $this->tvnSize;
}
for ($j = 0; $j < $size; $j++) {
$this->timerManager[$i][$j] = [];
}
}
$this->ary[3] = date('t');
$this->lastTime = time();
}
/**
* uExpires
* format a recently exec time, then make a expires time
* receive a frequency like crontab * * * * * *
* @param string $frequency
* @return number
*/
public function uExpires(string $frequency)
{
$fr = explode(" ", $frequency);
$fields = []; //列举多维度的有效时间数组
for ($i = 0; $i < count($fr); $i++) {
if ($fr[$i] == '*') {
$fields[$i] = range(0, $this->ary[$i] - 1);
} elseif (preg_match("/\*\/(\d*)/", $fr[$i], $matches)) {
$fields[$i] = ['f' => $matches[1]];
} elseif (preg_match("/(\d*)-(\d*)/", $fr[$i], $matches)) {
$fields[$i] = range($matches[1], $matches[2]);
} else {
$fields[$i] = explode(",", $fr[$i]);
}
}
//print_r($fields);
/* 查找一个最近的有效时间 */
$incr = 0;
$year = date("Y");
$tk = [0,0,0,0,0,0];//时间指针数组,代表查找到的每一个纬度最近时间
for ($i = 0; $i < count($fr); $i++) {
$d = date($this->fc[$i]);
if (!isset($fields[$i]['f'])) {
// 检查合适的时间位置
for ($j = 0; $j < count($fields[$i]); $j++) {
if (($d <= $fields[$i][$j] && $j == 0) || ($d <= $fields[$i][$j] && $d > $fields[$i][$j-1])) {
if ($incr) {
if ($fields[$i][$j] == $d) {
$j++;
}
//有进位则把前面的指针都置为零
foreach ($tk as $k => &$v) {
if (isset($fields[$k]['f'])) {
$fields[$k][0] = ($d + $fields[$k]['f']) % $this->ary[$k];
} else {
$v = 0;
}
}
}
break;
}
}
//echo $i,' ',$incr,' '.$j." ";
if (!isset($fields[$i][$j])) {
$j = 0;
if ($i == 4) {
$year++;
$incr = 0; //消除进位
//有进位则把前面的指针都置为零
foreach ($tk as $k => &$v) {
if (isset($fields[$k]['f'])) {
$fields[$k][0] = ($d + $fields[$k]['f']) % $this->ary[$k];
} else {
$v = 0;
}
}
} else {
$incr = 1; //产生进位
}
} else {
$incr = 0; //消除进位
}
$tk[$i] = $j;
} else {
if ($incr) {
$incr = 0;
//有进位则把前面的指针都置为零
foreach ($tk as $k => &$v) {
if (isset($fields[$k]['f'])) {
$fields[$k][0] = ($d + $fields[$k]['f']) % $this->ary[$k];
} else {
$v = 0;
}
}
$fields[$i][0] = $d + $fields[$i]['f'] + 1;
} else {
$fields[$i][0] = $d + $fields[$i]['f'];
}
if ($fields[$i][0] >= $this->ary[$i]) {
$incr = 1;
}
$fields[$i][0] %= $this->ary[$i];
}
}
//按照指针数组合成最近执行时间
$datetime = $year."-".$fields[4][$tk[4]]."-".$fields[3][$tk[3]]." ".$fields[2][$tk[2]].":".$fields[1][$tk[1]].":".$fields[0][$tk[0]];
$this->printLog(__FUNCTION__." ".date("Y-m-d H:i:s")." ".$frequency." => ".$datetime);
return (int)((strtotime($datetime) - (time() - $this->uJiffies))); //定时器到期时间
}
/**
* create a task
* @param unknown $frequency
* @param unknown $id
* @return unknown[]|number[]
*/
public function createTask($frequency, $id)
{
$this->printLog("-------------------------begin createTask---------------------------");
if (isset($this->arrTask[$id])) {
$this->arrTask[$id]['fre'] = $frequency;
$this->arrTask[$id]['uExpires'] = $this->uExpires($frequency);
} else {
$task = ['fre' => $frequency, 'id' => $id, 'uExpires' => $this->uExpires($frequency)];
$this->arrTask[$id] = $task;
}
$this->printLog(__FUNCTION__." ".json_encode($this->arrTask[$id]));
$this->addTask($this->arrTask[$id]);
$this->printLog("--------------------------end createTask----------------------------");
return $task;
}
/**
* add task
* @param unknown $task
*/
public function addTask($task)
{
$id = $task['id'];
$uExpires = $task['uExpires']; //距离定时器到期时间
$uDueTime = $uExpires - $this->uJiffies; //触发剩余时间
//$this->printLog(__FUNCTION__." ".$uDueTime." ".json_encode($this->arrTask));
if ($uDueTime < $this->tvrSize) {
$i = $uExpires & $this->tvrMask;
$this->timerManager[0][$i][$id] = &$this->arrTask[$id];
$this->arrTask[$id][1] = 0; //轮的层级
} elseif ($uDueTime < 1 << ($this->tvrBits + $this->tvnBits)) {
$i = ($uExpires >> $this->tvrBits) & $this->tvnMask;
$this->timerManager[1][$i][$id] = &$this->arrTask[$id];
$this->arrTask[$id][1] = 1;
} elseif ($uDueTime < 1 << ($this->tvrBits + $this->tvnBits * 2)) {
$i = ($uExpires >> ($this->tvrBits + $this->tvnBits)) & $this->tvnMask;
$this->timerManager[2][$i][$id] = &$this->arrTask[$id];
$this->arrTask[$id][1] = 2;
} elseif ($uDueTime < 1 << ($this->tvrBits + $this->tvnBits * 3)) {
$i = ($uExpires >> ($this->tvrBits + $this->tvnBits * 2)) & $this->tvnMask;
$this->timerManager[3][$i][$id] = &$this->arrTask[$id];
$this->arrTask[$id][1] = 3;
} elseif ($uDueTime < 1 << ($this->tvrBits + $this->tvnBits * 4)) {
$i = ($uExpires >> ($this->tvrBits + $this->tvnBits * 3)) & $this->tvnMask;
$this->timerManager[4][$i][$id] = &$this->arrTask[$id];
$this->arrTask[$id][1] = 4;
}
$this->arrTask[$id][2] = $i; //轮中的槽
$this->printLog(__FUNCTION__." ".json_encode($this->arrTask[$id])." ".$uDueTime);
//echo __FUNCTION__." ".json_encode($this->arrTask[$id])."\n";
}
public function delTask($id)
{
if (isset($this->arrTask[$id])) {
unset($this->timerManager[$this->arrTask[$id][1]][$this->arrTask[$id][2]][$id]);
unset($this->arrTask[$id]);
}
return 1;
}
public function updateTask($id)
{
if (isset($this->arrTask[$id])) {
unset($this->timerManager[$this->arrTask[$id][1]][$this->arrTask[$id][2]][$id]);
$this->createTask($this->arrTask[$id]['fre'], $this->arrTask[$id]['id']);
}
return 1;
}
/**
* cascade update task
* @param unknown $lv
*/
public function cascadeTask($lv)
{
if ($lv == 1) {
$i = $this->uJiffies >> $this->tvrBits;
} else {
$i = $this->uJiffies >> ($this->tvrBits + $this->tvnBits * ($lv) - 1);
}
if ($i > 0 && 0 == $i % $this->tvnSize) { //判断时间进位
$this->cascadeTask($lv+1);
}
$i %= $this->tvnSize;
$this->printLog("lv=".$lv." i=".$i);
foreach ($this->timerManager[$lv][$i] as $id => $task) {
unset($this->timerManager[$lv][$i][$id]);
$this->addTask($task);
}
}
public function getTask()
{
$taskList = [];
$now = time();
//$now = $this->lastTime+1;
if ($this->lastTime != $now) {
//$this->printLog("++++++++++++++++++++++++++++begin getTask++++++++++++++++++++++++++++");
while ($this->lastTime != $now) { //避免程序执行产生的时间误差导致跳过>1秒
if ($this->uJiffies != 0 && ($this->uJiffies % $this->tvrSize) == 0) {
$this->printLog("jiff=".$this->uJiffies);
$this->cascadeTask(1);
}
$taskList = array_merge($taskList, $this->timerManager[0][$this->uJiffies % $this->tvrMask]);
$this->uJiffies++; //瞬间时间++
$this->lastTime++;
foreach ($taskList as $task) {
$this->printLog(__FUNCTION__." ".json_encode($task)." ".($this->uJiffies-1));
$this->updateTask($task['id']);
}
}
//$this->printLog("+++++++++++++++++++++++++++++end getTask+++++++++++++++++++++++++++++");
}
return $taskList;
}
public function setLogFile($logFile)
{
$this->logFile = $logFile;
}
public function printLog($msg)
{
if (is_array($msg)) {
$msg = json_encode($msg);
}
$msg = date("Y-m-d H:i:s")." ".posix_getpid()." ".$msg."\n";
if ($this->logFile) {
file_put_contents($this->logFile, $msg, FILE_APPEND | LOCK_EX);
} else {
echo $msg;
}
}
}
$timer = new TimerWheel();
$timer->createTask("* 15 12-20 * * *", 333);
$timer->createTask("15 10-15 * * * *", 333);
$timer->createTask("*/10 * * * * *", 222);
$timer->createTask("15 */2 * * * *", 333);
$timer->createTask("25 14 */3 09 * *", 555);
$timer->createTask("25 10 */3 * * *", 555);
$timer->createTask("* * * * * *", 111);
$timer->createTask("10 * * * * *", 222);
$timer->createTask("15 10 * * * *", 333);
$timer->createTask("* 10,20,30 * * * *", 334);
$timer->createTask("20 12 08 * * *", 444);
$timer->createTask("* * 08 * * *", 445);
$timer->createTask("25 14 18 09 * *", 555);
$timer->createTask("* * * 09 * *", 556);
$timer->createTask("30 12 * * 1 *", 666);
$timer->createTask("* 12 * * 1 *", 666);
for ($i = 0; $i < 100; $i++) {
$timer->getTask();
}
转载请注明:小Y » php实现时间轮算法