溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

php沒有定時功能的原因是什么

發(fā)布時間:2023-02-24 10:18:13 來源:億速云 閱讀:110 作者:iii 欄目:編程語言

今天小編給大家分享一下php沒有定時功能的原因是什么的相關知識點,內容詳細,邏輯清晰,相信大部分人都還太了解這方面的知識,所以分享這篇文章給大家參考一下,希望大家閱讀完這篇文章后有所收獲,下面我們一起來了解一下吧。

web開發(fā)中PHP不能使用定時器的本質原因是可控常駐內存運行環(huán)境的缺失;兩個要點,第一常駐內存,第二可控。CGI模式下,進程執(zhí)行完腳本后直接退出,不能指望其到指定時間運行任務;PHP-FPM模式下,進程常駐內存,但不可控。

php為什么沒有定時功能

常見的定時器有兩種:一種周期性定時執(zhí)行,例如每天的凌晨三點出報表;另一種在指定時間后執(zhí)行(一次),例如會員登錄系統(tǒng)五分鐘后發(fā)放每日登錄獎勵。對應JavaScript中的setInterval和setTimeout函數(shù)(嚴格來說setInterval是周期性執(zhí)行,指定時間點執(zhí)行需要自行處理)。

做web開發(fā)的PHP程序員對JavaScript中的兩個定時器函數(shù)應該都還熟悉,回到PHP層面就有點傻眼:

PHP中有sleep,但是沒有(內置)定時器函數(shù)可用。sleep函數(shù)勉強可以做到,但會導致進程阻塞,期間不能做其他事(或無響應)。為什么PHP沒能提供定時器(Timer)這個功能呢?

原因

web開發(fā)中PHP不能使用定時器的本質原因是可控 常駐內存運行環(huán)境的缺失。兩個要點:第一常駐內存,第二可控。CGI模式下,進程執(zhí)行完腳本后直接退出,不能指望其到指定時間運行任務;PHP-FPM模式下,進程(絕大多數(shù))常駐內存,但不可控。

不可控的意思是執(zhí)行PHP的進程不受PHP代碼影響,進程的入口點和退出時機由額外的程序控制。例如FPM模式下,PHP腳本中的exit、die函數(shù)只中斷腳本的執(zhí)行,不會對執(zhí)行腳本的進程產生特別的影響(內存泄露除外)。PHP開發(fā)人員編寫的腳本是進程的執(zhí)行體,執(zhí)行完畢后就從進程的執(zhí)行上下文中卸載出去。這種情況下,執(zhí)行PHP腳本的時機仍然由外部驅動,沒有外部請求PHP代碼就安詳?shù)奶稍谟脖P上,什么都不做,也就定時任務。

由于PHP主要面向web開發(fā),PHP這種執(zhí)行模式穩(wěn)定可靠,開發(fā)效率快。比如省去資源釋放這一步,就避免了開發(fā)中很多工作量和坑。想想某些第三方庫代碼中改時區(qū)、字符編碼等還不還原,在常駐內存運行環(huán)境下幾乎肯定會導致后續(xù)請求有問題。但在FPM模式下,這種坑無意中直接趟平,省去許多調試時間,為程序員保住發(fā)際線做出了不小的貢獻。

問題已經了解,那么PHP中如何使用定時器執(zhí)行定時任務?

危險的做法

在web環(huán)境下,PHP腳本默認有超時時間。去掉超時設置,就可以讓程序一直在后臺運行(如果進程不退出的話)。例如以下代碼在響應請求后繼續(xù)后臺運行,并且每五秒鐘輸出一次時間到文件:

# test.php
set_time_limit(0); # 取消超時設置,讓腳本可一直運行

echo 'This is a background run forever script. Now you can leave me alone.';

fastcgi_finish_request();   # 結束當前請求

do{
   file_put_contents("/tmp/out.dat", "test script, now:" . date("Y-m-d H:i:s") . "\n", FILE_APPEND);
   sleep(5);
}while(true);

請求http://localhost:8080/test.php文件后,監(jiān)測/tmp/out.dat文件,會發(fā)現(xiàn)不斷有內容輸出,無論客戶端是否斷開連接、關閉瀏覽器或者重啟電腦(不能重啟服務器)。這說明程序一直在執(zhí)行,并且也實現(xiàn)了我們想要的定時器功能。如果把sleep改成usleep、time_nanosleep,還能實現(xiàn)微秒、納秒級定時器,豈不美哉?

實踐中應當盡量避免用這種方式實現(xiàn)定時器,不僅因為低效,還略有危險。原因之一是每次請求會占用一個進程,請求十萬次需要十萬個進程,基本上會導致系統(tǒng)崩潰或后續(xù)請求無響應;另外如果打開了session,但是忘記調用session_write_close,會導致同一個用戶的后續(xù)請求被hang?。╯ession活躍時處于加鎖狀態(tài),不關閉session會導致后續(xù)進程無法打開session)。

web開發(fā)應當越快響應用戶的請求越好,在web開發(fā)中用這種方式強行實現(xiàn)定時器,會讓整個web應用處于不穩(wěn)定、不可靠或不可預測狀態(tài)。孟子曰:知而慎行,君子不立于危墻之下。不靠譜的做法要盡量避免,順帶也避免背鍋和甩鍋。

接下來看看PHP中使用定時器的正確姿勢。

正確的姿勢

PHP實現(xiàn)定時器的做法可簡單歸結為如下幾種:

  • 使用cron、Jenkins等調度工具做周期性定時任務(既可以是執(zhí)行腳本,也可以是請求某個網址);

  • 一次性執(zhí)行任務通過消息隊列、數(shù)據(jù)庫等方式投遞給第三方程序執(zhí)行;

  • 像WordPress一樣模擬定時任務,但要記住這種方式依賴于客戶端請求,并需自行處理好進程并發(fā)問題;

  • 使用常駐內存型方式運行PHP程序,即CLI模式。

除了第三種做法,其他方式都是推薦的,具體方案請結合實際需求。作為PHP程序員,當然還是首選用PHP來做,也就是CLI模式。

CLI模式

摸著良心說,CLI模式讓PHP發(fā)揮的空間拓展不少。在CLI模式下,程序的入口點就是腳本,且代碼可以常駐內存,進程完全由PHP代碼控制。在這種形式下,實現(xiàn)定時器就有多種玩法。本文列出幾種做法,拋磚引玉:

  • 使用swoole、workerman等框架,內置(高精度)定時器;

  • 使用多進程(池)/多線程(池)技術(pcntl、pthreads拓展在CLI模式下才可用);

  • 處理tick或者alarm等信號;

  • 使用libevent、libev等事件驅動庫;

  • sleep加循環(huán)或自己實現(xiàn)事件循環(huán)。

想折騰的話自己用2-5方案,不想折騰swoole、workerman等框架是首選,穩(wěn)定可靠。

以上就是“php沒有定時功能的原因是什么”這篇文章的所有內容,感謝各位的閱讀!相信大家閱讀完這篇文章都有很大的收獲,小編每天都會為大家更新不同的知識,如果還想學習更多的知識,請關注億速云行業(yè)資訊頻道。

向AI問一下細節(jié)

免責聲明:本站發(fā)布的內容(圖片、視頻和文字)以原創(chuàng)、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關證據(jù),一經查實,將立刻刪除涉嫌侵權內容。

php
AI