常見的計時器有兩種:一種週期性定時執行,例如每天的凌晨三點出報表;另一種在指定時間後執行(一次),例如會員登入系統五分鐘後發放每日登入獎勵。兩種情況對應shell中的cron和at指令,與JavaScript中的setInterval和setTimeout函數類似(嚴格來說setInterval是週期性執行,指定時間點執行需要自行處理)。
做web開發的PHP程式設計師對JavaScript中的兩個計時器函數應該都還熟悉,回到PHP層面就有點傻眼:
PHP中有sleep,但沒有(內建)計時器函數可用。 sleep函數勉強可以做到,但會導致進程阻塞,期間不能做其他事(或無回應)。為什麼PHP沒能提供定時器(Timer)這個功能呢?
原因
個人認為,web開發中PHP不能使用定時器的本質原因是可控 常駐記憶體運行環境的缺失。兩個重點:第一常駐內存,第二可控。 CGI模式下,進程執行完腳本後直接退出,不能指望其到指定時間運行任務;PHP-FPM模式下,進程(絕大多數)常駐內存,但不可控。
不可控的意思是執行PHP的進程不受PHP程式碼影響,進程的入口點和退出時機由額外的程式控制。例如FPM模式下,PHP腳本中的exit、die函數只會中斷腳本的執行,不會對執行腳本的程序產生特別的影響(記憶體外洩除外)。 PHP開發人員編寫的腳本是進程的執行體,執行完畢後就從進程的執行上下文中卸載出去。在這種情況下,執行PHP腳本的時機仍然由外部驅動,沒有外部請求PHP程式碼就安詳的躺在硬碟上,什麼都不做,也就定時任務。
由於PHP主要針對web開發,PHP這種執行模式穩定可靠,開發效率快。例如省去資源釋放這一步,就避免了開發中許多工作量和坑。想想某些第三方函式庫程式碼中改時區、字元編碼等還不還原,在常駐記憶體運作環境下幾乎肯定會導致後續請求有問題。但在FPM模式下,這種坑無意中直接趟平,省去許多調試時間,為程式設計師保住髮際線做出了不小的貢獻。
以上是php為什麼不支援定時器?的詳細內容。更多資訊請關注PHP中文網其他相關文章!