站長資訊網(wǎng)
最全最豐富的資訊網(wǎng)站

PHP多任務秒級定時器的實現(xiàn)方法

描述

最近在公司部署crontab的時候,突發(fā)奇想是否可以用PHP去實現(xiàn)一個定時器,顆粒度到秒級就好,因為crontab最多到分鐘級別,同時也調(diào)研了一下用PHP去實現(xiàn)的定時器還真不太多,Swoole 擴展里面到實現(xiàn)了一個毫秒級的定時器很高效,但畢竟不是純PHP代碼寫的,所以最后還是考慮用PHP去實現(xiàn)一個定時器類,以供學習參考。

實現(xiàn)

在實現(xiàn)定時器代碼的時候,用到了PHP系統(tǒng)自帶的兩個擴展

Pcntl – 多進程擴展 :

主要就是讓PHP可以同時開啟很多子進程,并行的去處理一些任務。

Spl – SplMinHeap – 小頂堆

一個小頂堆數(shù)據(jù)結(jié)構(gòu),在實現(xiàn)定時器的時候,采用這種結(jié)構(gòu)效率還是不錯的,插入、刪除的時間復雜度都是 O(logN) ,像 libevent 的定時器也在 1.4 版本以后采用了這種數(shù)據(jù)結(jié)構(gòu)之前用的是 rbtree,如果要是使用鏈表或者固定的數(shù)組,每次插入、刪除可能都需要重新遍歷或者排序,還是有一定的性能問題的。

流程

PHP多任務秒級定時器的實現(xiàn)方法

說明

1、定義定時器結(jié)構(gòu),有什么參數(shù)之類的.

2、然后全部注冊進我們的定時器類 Timer.

3、調(diào)用定時器類的monitor方法,開始進行監(jiān)聽.

4、監(jiān)聽過程就是一個while死循環(huán),不斷的去看時間堆的堆頂是否到期了,本來考慮每秒循環(huán)看一次,后來一想每秒循環(huán)看一次還是有點問題,如果正好在我們sleep(1)的時候定時器有到期的了,那我們就不能馬上去精準執(zhí)行,可能會有延時的風險,所以還是采用 usleep(1000) 毫秒級的去看并且也可以將進程掛起減輕 CPU 負載.

代碼

/*** * Class Timer */ class Timer extends SplMinHeap {   /**   * 比較根節(jié)點和新插入節(jié)點大小   * @param mixed $value1   * @param mixed $value2   * @return int   */   protected function compare($value1, $value2)   {     if ($value1['timeout'] > $value2['timeout']) {       return -1;     }     if ($value1['timeout'] < $value2['timeout']) {       return 1;     }     return 0;   }   /**   * 插入節(jié)點   * @param mixed $value   */   public function insert($value)   {     $value['timeout'] = time() + $value['expire'];     parent::insert($value);   }   /**   * 監(jiān)聽   * @param bool $debug   */   public function monitor($debug = false)   {     while (!$this->isEmpty()) {       $this->exec($debug);       usleep(1000);     }   }   /**   * 執(zhí)行   * @param $debug   */   private function exec($debug)   {     $hit = 0;     $t1  = microtime(true);     while (!$this->isEmpty()) {       $node = $this->top();       if ($node['timeout'] <= time()) {         //出堆或入堆         $node['repeat'] ? $this->insert($this->extract()) : $this->extract();         $hit = 1;         //開啟子進程         if (pcntl_fork() == 0) {           empty($node['action']) ? '' : call_user_func($node['action']);           exit(0);         }         //忽略子進程,子進程退出由系統(tǒng)回收         pcntl_signal(SIGCLD, SIG_IGN);       } else {         break;       }     }     $t2 = microtime(true);     echo ($debug && $hit) ? '時間堆 - 調(diào)整耗時: ' . round($t2 - $t1, 3) . "秒rn" : '';   } }

實例

$timer = new Timer(); //注冊 - 3s - 重復觸發(fā) $timer->insert(array('expire' => 3, 'repeat' => true, 'action' => function(){   echo '3秒 - 重復 - hello world' . "rn"; })); //注冊 - 3s - 重復觸發(fā) $timer->insert(array('expire' => 3, 'repeat' => true, 'action' => function(){   echo '3秒 - 重復 - gogo' . "rn"; })); //注冊 - 6s - 觸發(fā)一次 $timer->insert(array('expire' => 6, 'repeat' => false, 'action' => function(){   echo '6秒 - 一次 - hello xxxx' . "rn"; })); //監(jiān)聽 $timer->monitor(false);

執(zhí)行結(jié)果

PHP多任務秒級定時器的實現(xiàn)方法

也測試過比較極端的情況,同時1000個定時器1s全部到期,時間堆全部調(diào)整完僅需 0.126s 這是沒問題的,但是每調(diào)整完一個定時器就需要去開啟一個子進程,這塊可能比較耗時了,有可能1s處理不完這1000個,就會影響下次監(jiān)聽繼續(xù)觸發(fā),但是不開啟子進程,比如直接執(zhí)行應該還是可以處理完的。。。。當然肯定有更好的方法,目前只能想到這樣。

贊(0)
分享到: 更多 (0)
網(wǎng)站地圖   滬ICP備18035694號-2    滬公網(wǎng)安備31011702889846號
99视频精品全部在线观看| 国产精品兄妹在线观看麻豆| 精品国产精品国产| 亚洲伊人久久精品| 久久se精品一区二区国产 | 亚洲乱码精品久久久久..| 婷婷国产成人精品一区二| 日韩土豪美女在线视频观看| 思思99re66在线精品免费观看| 2020国产精品自拍| 99精品众筹模特自拍视频| 久久精品国产影库免费看| 东京热TOKYO综合久久精品| 国产精品99爱免费视频| 日韩免费福利视频| 无码日韩人妻精品久久蜜桃| 国产精品一区二区AV麻豆| 在线精品免费视频| 亚洲国产高清国产拍精品| 69久久精品无码一区二区| 久久人人做人人玩人精品| 在线精品91青草国产在线观看| 亚洲午夜精品久久久久久人妖| 久久99精品综合国产首页| 野狼第一精品社区| 自拍偷自拍亚洲精品第1页 | 国产伦精品一区二区三区| 中国精品一级毛片免费播放| 国产内地精品毛片视频| 亚洲国产成人精品无码久久久久久综合 | 亚洲国产精品久久久久| 久久精品乱子伦免费| 久久精品亚洲综合| 亚洲成人精品久久| 99re最新地址精品视频 | 四虎永久精品免费观看| 国产精品亚洲а∨天堂2021| 成人免费无码精品国产电影| 国产精品无码一本二本三本色| 国产精品爆乳奶水无码视频| 国产精品99re|