我們在循環執行某個程序時,可能會出現超時導致程序死掉的情況。所以我們有必要限制每個循環執行的最長時間,以此來避免程序死掉的情況。
如果超時,則直接斷開改進程,并繼續下一層循環操作。攜程,多線程都可以完成該操作,但在沒有了解這些高深技術的時候,可以用這個簡便的方法替代。
TaskAsync.php
namespace TaskAsync; use WorkermanMySQLConnection; class TaskAsync { /** * 異步任務 * @params $func 要異步執行的主要函數 * @params $func 要異步執行超時后的函數 * @params $maxTime 異步執行超時的時間 單位:秒 s * @params $params 要傳遞給$func的參數 */ public static function asyncTask(callable $func, $params = array(), $maxTime = 0, callable $func2 = null, $params2 = array()){ pcntl_signal(SIGCHLD, SIG_IGN); //安裝監聽信號 $pid = pcntl_fork(); //生成一個線程 if ($pid == -1) { exit();//創建子進程失敗 } else if ($pid == 0) { //邏輯 try { //執行用戶函數 call_user_func_array($func, $params); } finally { //執行完后殺死進程 posix_kill(posix_getpid(), SIGKILL); exit(0);//結束子進程的操作 } } else if ($pid > 0) { $t = time(); while (true) { $nPid= pcntl_wait($s, WNOHANG); if ($nPid > 0) { break; } else if ($nPid < 0) { break; } else if ($maxTime && time() - $t > $maxTime) { //默認超時時間為0 ,即 不限制超時時間 posix_kill($pid, SIGKILL); if (!empty($func2)) { call_user_func_array($func2, $params2); } break; } else { sleep(1);//每秒輪詢檢查 } } } } public static function getMysqlConn() { $dbConfig = require(APP_PATH . '/database.php'); return new Connection($dbConfig['hostname'], $dbConfig['hostport'], $dbConfig['username'], $dbConfig['password'], $dbConfig['database']); } }
(學習視頻分享:php視頻教程)
index.php
use TaskAsyncTaskAsync; //使用 while(true) { $db = TaskAsync::getMysqlConn(); //數據庫操作 $db->closeConnection(); $data = [] ;//傳入的數據 TaskAsync::asyncTask(array(new Download(),'downLoadExcel'), array($data), 60 * 60 * 10 , function($data){ echo '執行超時' ; }, array($data) ); }
pcntl_fork 之前,不能有數據庫連接操作,所以,如果涉及到數據庫的操作,在每次循環的時候,一定要重新連接數據庫,執行完操作,要記得斷開數據庫連接,否則這樣會提示 MySQL server has gone away !