《PHP實(shí)戰(zhàn):詳解PHP執(zhí)行定時(shí)任務(wù)的實(shí)現(xiàn)思路》要點(diǎn):
本文介紹了PHP實(shí)戰(zhàn):詳解PHP執(zhí)行定時(shí)任務(wù)的實(shí)現(xiàn)思路,希望對(duì)您有用。如果有疑問(wèn),可以聯(lián)系我們。
PHP本身是沒(méi)有定時(shí)功能的,PHP也不能多線程.PHP的定時(shí)任務(wù)功能必須通過(guò)和其他工具結(jié)合才能實(shí)現(xiàn),例如WordPress內(nèi)置了wp-cron的功能,很厲害.PHP實(shí)例
一、Linux服務(wù)器上使用CronTab定時(shí)執(zhí)行phpPHP實(shí)例
我們先從相對(duì)比較復(fù)雜的服務(wù)器執(zhí)行php談起.服務(wù)器上安裝了php,就可以執(zhí)行php文件,無(wú)論是否安裝了nginx或Apache這樣的服務(wù)器環(huán)境軟件.而Linux中,使用命令行,用CronTab來(lái)定時(shí)任務(wù),又是絕佳的選擇,而且也是效率最高的選擇.PHP實(shí)例
首先,進(jìn)入命令行模式.作為服務(wù)器的linux一般都默認(rèn)進(jìn)入命令行模式的,當(dāng)然,我們管理服務(wù)器也一般通過(guò)putty等工具遠(yuǎn)程連接到服務(wù)器,為了方便,我們用root用戶登錄.在命令行中鍵入:PHP實(shí)例
crontab -e
之后就會(huì)打開(kāi)一個(gè)文件,并且是非編輯狀態(tài),則是vi的編輯界面,通過(guò)敲鍵盤上的i,進(jìn)入編輯模式,就可以編輯內(nèi)容.這個(gè)文件中的每一行就是一個(gè)定時(shí)任務(wù),我們新建一行,就是新建一條定時(shí)任務(wù)(當(dāng)然是指這一行內(nèi)依照一定的格式進(jìn)行書(shū)寫).我們現(xiàn)在來(lái)舉個(gè)例子,增加一行,內(nèi)容如下:PHP實(shí)例
00 * * * * lynx -dump https://www.yourdomain.com/script.php
這是什么意思呢?實(shí)際上上面這一行由兩部分組成,前面一部分是時(shí)間,后面一部分是操作內(nèi)容.例如上面這個(gè),PHP實(shí)例
00 * * * *
就是指當(dāng)當(dāng)前時(shí)間的分鐘數(shù)為00時(shí),執(zhí)行該定時(shí)任務(wù).時(shí)間部分由5個(gè)時(shí)間參數(shù)組成,分別是:PHP實(shí)例
分 時(shí) 日 月 周
第1列表示分鐘1~59 每分鐘用或者 */1表示,/n表示每n分鐘,例如*/8就是每8分鐘的意思,下面也是類推
第2列表示小時(shí)1~23(0表示0點(diǎn))
第3列表示日期1~31
第4列表示月份1~12
第5列標(biāo)識(shí)號(hào)星期0~6(0表示星期天)PHP實(shí)例
整個(gè)句子的后面部分就是操作的具體內(nèi)容.PHP實(shí)例
lynx -dump https://www.yourdomain.com/script.php
意思就是說(shuō)通過(guò)lynx拜訪這個(gè)url.我們?cè)谑褂弥兄饕玫絣ynx、curl、wget來(lái)實(shí)現(xiàn)對(duì)url的遠(yuǎn)程拜訪,而如果要提高效率,直接用php去執(zhí)行本地php文件是最佳選擇,例如:PHP實(shí)例
00 */2 * * * /usr/local/bin/php /home/www/script.php
這條語(yǔ)句就可以在每2小時(shí)的0分鐘,通過(guò)linux內(nèi)部php環(huán)境執(zhí)行script.php,注意,這里可不是通過(guò)url拜訪,通過(guò)服務(wù)器環(huán)境來(lái)執(zhí)行哦,而是直接執(zhí)行,因?yàn)槔@過(guò)了服務(wù)器環(huán)境,所以效率當(dāng)然要高很多.PHP實(shí)例
好了,已經(jīng)添加了幾條需要的定時(shí)任務(wù)了吧.點(diǎn)擊鍵盤上的Esc鍵,輸入“:wq”回車,這樣就保存了設(shè)置的定時(shí)任務(wù),屏幕上也能看到提示創(chuàng)建了新的定時(shí)任務(wù).接下來(lái)就是好好寫你的script.php了.PHP實(shí)例
關(guān)于CronTab的更多用法這里就不介紹了,如果你想更靈活的使用這個(gè)定時(shí)任務(wù)功能,應(yīng)該自己再去深入學(xué)習(xí)一下crontab.PHP實(shí)例
二、Windows服務(wù)器上使用bat定時(shí)執(zhí)行phpPHP實(shí)例
windows上和linux上有一個(gè)類似的cmd和bat文件,bat文件類似于shell文件,執(zhí)行這個(gè)bat文件,就相當(dāng)于依次執(zhí)行里面的命令(當(dāng)然,還可以通過(guò)邏輯來(lái)實(shí)現(xiàn)編程),所以,我們可以利用bat命令文件在windows服務(wù)器上面實(shí)現(xiàn)PHP定時(shí)任務(wù).實(shí)際上在windows上定時(shí)任務(wù),和linux上道理是一樣的,只不過(guò)辦法和途徑不同.好了下面開(kāi)始.PHP實(shí)例
首先,在一個(gè)你覺(jué)得比較適當(dāng)?shù)奈恢脛?chuàng)建一個(gè)cron.bat文件,然后用文本編輯器打開(kāi)它(記事本都可以),在里面寫上這樣的內(nèi)容:PHP實(shí)例
D:\php\php.exe -q D:\website\test.php
這句話的意思就是,使用php.exe去執(zhí)行test.php這個(gè)php文件,和上面的contab一樣,繞過(guò)了服務(wù)器環(huán)境,執(zhí)行效率也比較高.寫好之后,點(diǎn)擊保存,關(guān)閉編輯器.PHP實(shí)例
接下來(lái)就是設(shè)置定時(shí)任務(wù)來(lái)運(yùn)行cron.bat.依次打開(kāi):“開(kāi)始C>控制面板C>任務(wù)計(jì)劃C>添加任務(wù)計(jì)劃”,在打開(kāi)的界面中設(shè)置定時(shí)任務(wù)的時(shí)間、暗碼,通過(guò)選擇,把cron.bat掛載進(jìn)去.確定,這樣一個(gè)定時(shí)任務(wù)就建立好了,在這個(gè)定時(shí)任務(wù)上右鍵,運(yùn)行,這個(gè)定時(shí)任務(wù)就開(kāi)始執(zhí)行了,到點(diǎn)時(shí),就會(huì)運(yùn)行cron.bat處理,cron.bat再去執(zhí)行php.PHP實(shí)例
三、非自有服務(wù)器(虛擬主機(jī))上實(shí)現(xiàn)php定時(shí)任務(wù)PHP實(shí)例
如果站長(zhǎng)沒(méi)有自己的服務(wù)器,而是租用虛擬主機(jī),就無(wú)法進(jìn)入服務(wù)器系統(tǒng)進(jìn)行上述操作.這個(gè)時(shí)候應(yīng)該如何進(jìn)行php定時(shí)任務(wù)呢?其實(shí)辦法又有多個(gè).PHP實(shí)例
1、使用ignore_user_abort(true)和sleep死循環(huán)PHP實(shí)例
在一個(gè)php文檔的開(kāi)頭直接來(lái)一句:PHP實(shí)例
ignore_user_abort(true);
這時(shí),通過(guò)url拜訪這個(gè)php的時(shí)候,即使用戶把瀏覽器關(guān)掉(斷開(kāi)連接),php也會(huì)在服務(wù)器上繼續(xù)執(zhí)行.利用這個(gè)特性,我們可以實(shí)現(xiàn)非常牛的功能,也就是通過(guò)它來(lái)實(shí)現(xiàn)定時(shí)任務(wù)的激活,激活之后就隨便它自己怎么辦了,實(shí)際上就有點(diǎn)類似于后臺(tái)任務(wù).PHP實(shí)例
而sleep(n)則是指當(dāng)程序執(zhí)行到這里時(shí),暫時(shí)不往下執(zhí)行,而是休息n秒鐘.如果你拜訪這個(gè)php,就會(huì)發(fā)現(xiàn)頁(yè)面起碼要加載n秒鐘.實(shí)際上,這種長(zhǎng)時(shí)間等待的行為是比較消耗資源的,不能大量使用.PHP實(shí)例
那么定時(shí)任務(wù)到底怎么實(shí)現(xiàn)呢?使用下面的代碼即可實(shí)現(xiàn):PHP實(shí)例
<?php ignore_user_abort(true); set_time_limit(0); date_default_timezone_set('PRC'); // 切換到中國(guó)的時(shí)間 $run_time = strtotime('+1 day'); // 定時(shí)任務(wù)第一次執(zhí)行的時(shí)間是明天的這個(gè)時(shí)候 $interval = 3600*12; // 每12個(gè)小時(shí)執(zhí)行一次 if(!file_exists(dirname(__FILE__).'/cron-run')) exit(); // 在目錄下存放一個(gè)cron-run文件,如果這個(gè)文件不存在,說(shuō)明已經(jīng)在執(zhí)行過(guò)程中了,該任務(wù)就不能再激活,執(zhí)行第二次,否則這個(gè)文件被多次拜訪的話,服務(wù)器就要崩潰掉了 do { if(!file_exists(dirname(__FILE__).'/cron-switch')) break; // 如果不存在cron-switch這個(gè)文件,就停止執(zhí)行,這是一個(gè)開(kāi)關(guān)的作用 $gmt_time = microtime(true); // 當(dāng)前的運(yùn)行時(shí)間,精確到0.0001秒 $loop = isset($loop) && $loop ? $loop : $run_time - $gmt_time; // 這里處理是為了確定還要等多久才開(kāi)始第一次執(zhí)行任務(wù),$loop就是要等多久才執(zhí)行的時(shí)間間隔 $loop = $loop > 0 ? $loop : 0; if(!$loop) break; // 如果循環(huán)的間隔為零,則停止 sleep($loop); // ... // 執(zhí)行某些代碼 // ... @unlink(dirname(__FILE__).'/cron-run'); // 這里就是通過(guò)刪除cron-run來(lái)告訴程序,這個(gè)定時(shí)任務(wù)已經(jīng)在執(zhí)行過(guò)程中,不能再執(zhí)行一個(gè)新的同樣的任務(wù) $loop = $interval; } while(true);
通過(guò)執(zhí)行上面這段php代碼,即可實(shí)現(xiàn)定時(shí)任務(wù),直到你刪除cron-switch文件,這個(gè)任務(wù)才會(huì)停止.PHP實(shí)例
但是有一個(gè)問(wèn)題,也就是如果用戶直接拜訪這個(gè)php,實(shí)際上沒(méi)有任何作用,頁(yè)面也會(huì)停在這個(gè)地方,一直處于加載狀態(tài),有沒(méi)有一種辦法可以消除這種影響呢?fsockopen幫我們解決了這個(gè)問(wèn)題.PHP實(shí)例
fsockopen可以實(shí)現(xiàn)在哀求訪問(wèn)某個(gè)文件時(shí),不必獲得返回結(jié)果就繼續(xù)往下執(zhí)行程序,這是和curl通常用法不一樣的地方,我們?cè)谑褂胏url訪問(wèn)網(wǎng)頁(yè)時(shí),一定要等curl加載完網(wǎng)頁(yè)后,才會(huì)執(zhí)行curl后面的代碼,雖然實(shí)際上curl也可以實(shí)現(xiàn)“非阻塞式”的哀求,但是比f(wàn)sockopen復(fù)雜的多,所以我們優(yōu)先選擇fsockopen,fsockopen可以在規(guī)定的時(shí)間內(nèi),比如1秒鐘以內(nèi),完成對(duì)訪問(wèn)路徑發(fā)出哀求,完成之后就不管這個(gè)路徑是否返回內(nèi)容了,它的任務(wù)就到這里結(jié)束,可以繼續(xù)往下執(zhí)行程序了.利用這個(gè)特性,我們?cè)谡5某绦蛄髦屑尤雈sockopen,對(duì)上面我們創(chuàng)建的這個(gè)定時(shí)任務(wù)php的地址發(fā)出哀求,即可讓定時(shí)任務(wù)在后臺(tái)執(zhí)行.如果上面這個(gè)php的url地址是www.yourdomain.com/script.php,那么我們?cè)诰幊讨?可以這樣:PHP實(shí)例
// ... // 正常的php執(zhí)行程序 // .. // 遠(yuǎn)程哀求(不獲取內(nèi)容)函數(shù),下面可以反復(fù)使用 function _sock($url) { $host = parse_url($url,PHP_URL_HOST); $port = parse_url($url,PHP_URL_PORT); $port = $port ? $port : 80; $scheme = parse_url($url,PHP_URL_SCHEME); $path = parse_url($url,PHP_URL_PATH); $query = parse_url($url,PHP_URL_QUERY); if($query) $path .= '?'.$query; if($scheme == 'https') { $host = 'ssl://'.$host; } $fp = fsockopen($host,$port,$error_code,$error_msg,1); if(!$fp) { return array('error_code' => $error_code,'error_msg' => $error_msg); } else { stream_set_blocking($fp,true);//開(kāi)啟了手冊(cè)上說(shuō)的非阻塞模式 stream_set_timeout($fp,1);//設(shè)置超時(shí) $header = "GET $path HTTP/1.1\r\n"; $header.="Host: $host\r\n"; $header.="Connection: close\r\n\r\n";//長(zhǎng)連接關(guān)閉 fwrite($fp, $header); usleep(1000); // 這一句也是關(guān)鍵,如果沒(méi)有這延時(shí),可能在nginx服務(wù)器上就無(wú)法執(zhí)行成功 fclose($fp); return array('error_code' => 0); } } _sock('www.yourdomain.com/script.php'); // ... // 繼續(xù)執(zhí)行其他動(dòng)作 // ..
把這段代碼加入到某個(gè)定時(shí)任務(wù)提交結(jié)果程序中,在設(shè)置好時(shí)間后,提交,然后執(zhí)行上面這個(gè)代碼,就可以激活該定時(shí)任務(wù),而且對(duì)于提交的這個(gè)用戶而言,沒(méi)有任何頁(yè)面上的堵塞感.PHP實(shí)例
2、借用用戶的拜訪行為來(lái)執(zhí)行某些延遲任務(wù)PHP實(shí)例
但是上面使用sleep來(lái)實(shí)現(xiàn)定時(shí)任務(wù),是效率很低的一種方案.我們希望不要使用這種方式來(lái)執(zhí)行,這樣的話就可以解決效率問(wèn)題.我們借用用戶拜訪行為來(lái)執(zhí)行任務(wù).用戶對(duì)網(wǎng)站的拜訪其實(shí)是一個(gè)非常豐富的行為資源,包括搜索引擎蜘蛛對(duì)網(wǎng)站的拜訪,都可以算作這個(gè)類型.在用戶拜訪網(wǎng)站時(shí),內(nèi)部加一個(gè)動(dòng)作,去檢查任務(wù)列表中是否存在沒(méi)有被執(zhí)行的任務(wù),如果存在,就將這個(gè)任務(wù)執(zhí)行.對(duì)于用戶而言,利用上面所說(shuō)的fsockopen,根本感覺(jué)不到自己的拜訪竟然還做出了這樣的貢獻(xiàn).但是這種拜訪的缺點(diǎn)就是拜訪很不規(guī)律,比如你希望在凌晨2點(diǎn)執(zhí)行某項(xiàng)任務(wù),但是這個(gè)時(shí)間段非常倒霉,沒(méi)有用戶或任何行為到達(dá)你的網(wǎng)站,直到早上6點(diǎn)才有一個(gè)新拜訪.這就導(dǎo)致你原本打算2點(diǎn)執(zhí)行的任務(wù),到6點(diǎn)才被執(zhí)行.PHP實(shí)例
這里涉及到一個(gè)定時(shí)任務(wù)列表,也就是說(shuō)你需要有一個(gè)列表來(lái)記錄所有任務(wù)的時(shí)間、執(zhí)行什么內(nèi)容.一般來(lái)說(shuō),很多系統(tǒng)會(huì)采用數(shù)據(jù)庫(kù)來(lái)記錄這些任務(wù)列表,比如wordpress就是這樣做的.我則利用文件讀寫特性,提供了托管在github上的開(kāi)源項(xiàng)目php-cron,你可以去看看.總之,如果你想要管理多個(gè)定時(shí)任務(wù),靠上面的單個(gè)php是無(wú)法合理布局的,必須想方法構(gòu)建一個(gè)schedules列表.由于這里面的邏輯比較復(fù)雜,就不再詳細(xì)闡述,我們僅停留在思路層面上.PHP實(shí)例
3、借用第三方定時(shí)任務(wù)跳板PHP實(shí)例
很好玩的是,一些服務(wù)商提供了各種類型的定時(shí)任務(wù),例如阿里云的ACE提供了單獨(dú)的定時(shí)任務(wù),你可以填寫自己應(yīng)用下的某個(gè)uri.百度云BCE提供了服務(wù)器監(jiān)測(cè)功能,每天會(huì)按照一定的時(shí)間規(guī)律訪問(wèn)應(yīng)用下的固定uri.類似的第三方平臺(tái)上還有很多定時(shí)任務(wù)可以用.你完全可以用這些第三方定時(shí)任務(wù)作為跳板,為你的網(wǎng)站定時(shí)任務(wù)服務(wù).比如說(shuō),你可以在阿里云ACE上建立一個(gè)每天凌晨2點(diǎn)的定時(shí)任務(wù),執(zhí)行的uri是/cron.php.然后你創(chuàng)建一個(gè)cron.php,里面則采用fsockopen去訪問(wèn)你真正要執(zhí)行某些任務(wù)的網(wǎng)站的url,例如上面的www.yourdomain.com/script.php,而且在cron.php中還可以訪問(wèn)多個(gè)url.然后把cron.php上傳到你的ACE上面去,讓ACE的定時(shí)任務(wù)去訪問(wèn)/cron.php,然后讓cron.php去遠(yuǎn)程哀求目標(biāo)網(wǎng)站的定時(shí)任務(wù)腳本.PHP實(shí)例
4、循環(huán)利用include包含文件(待驗(yàn)證)PHP實(shí)例
php面向過(guò)程的特性使得其程序是從上往下執(zhí)行的,利用這個(gè)特性,在我們使用include某個(gè)文件時(shí),就會(huì)執(zhí)行被引入的文件,知道include的文件內(nèi)程序執(zhí)行完之后,再往下執(zhí)行.如果我們創(chuàng)建一個(gè)循環(huán),再利用sleep,不斷的include某個(gè)文件,使循環(huán)執(zhí)行某段程序,則可以達(dá)到定時(shí)執(zhí)行的目的.我們?cè)龠M(jìn)一步,并不是利用while(true)來(lái)實(shí)現(xiàn)循環(huán),而是利用被include文件本身再include自身來(lái)實(shí)現(xiàn)循環(huán),比如我們創(chuàng)建一個(gè)do.php,它的內(nèi)容如下:PHP實(shí)例
if(...) exit(); // 通過(guò)某個(gè)開(kāi)關(guān)來(lái)關(guān)閉執(zhí)行 // ... // 執(zhí)行某些程序 // ... sleep($loop); // 這個(gè)$loop在include('do.php');之前賦值 include(dirname(__FILE__).'/do.php');
其實(shí)通過(guò)這種辦法執(zhí)行和while的思路也像.而且同樣用到sleep,效率低.PHP實(shí)例
PHP定時(shí)任務(wù)是一個(gè)非常有意思的東西,雖然說(shuō)實(shí)話,用系統(tǒng)的php.exe去直接執(zhí)行php文件的效率更高,但是對(duì)于很多普通站長(zhǎng)而言,虛擬主機(jī)是無(wú)法做到直接php執(zhí)行原生程序的.本文僅提供一些辦理的思路,我也僅僅是在學(xué)習(xí)中,有很多問(wèn)題或表述都不正確,希望你指出來(lái);你可以通過(guò)本文的思路,開(kāi)發(fā)出自己的一種辦理方案.PHP實(shí)例
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所贊助.PHP實(shí)例
歡迎參與《PHP實(shí)戰(zhàn):詳解PHP執(zhí)行定時(shí)任務(wù)的實(shí)現(xiàn)思路》討論,分享您的想法,維易PHP學(xué)院為您提供專業(yè)教程。
轉(zhuǎn)載請(qǐng)注明本頁(yè)網(wǎng)址:
http://www.fzlkiss.com/jiaocheng/8197.html