PHP 事件循環:異步編程的利器
核心要點
PHP 開發人員總是在等待某些事情。有時我們等待遠程服務的請求。有時我們等待數據庫返回來自複雜查詢的行。如果我們可以在所有等待期間執行其他操作,那不是很好嗎?
如果您編寫過一些 JS 代碼,您可能熟悉回調和 DOM 事件。雖然我們在 PHP 中也有回調,但它們的工作方式並不完全相同。這要歸功於一個稱為事件循環的功能。
我們將了解事件循環的工作原理,以及如何在 PHP 中使用事件循環。
我們將看到一些有趣的 PHP 庫。有些人認為這些庫還不夠穩定,無法用於生產環境。有些人認為此處提供的示例“最好在更成熟的語言中實現”。嘗試這些方法有很多很好的理由。在生產環境中避免這些方法也有很好的理由。這篇文章的目的是突出 PHP 中的可能性。
要理解事件循環,讓我們看看它們在瀏覽器中的工作方式。看看這個例子:
function fitToScreen(selector) { var element = document.querySelector(selector); var width = element.offsetWidth; var height = element.offsetHeight; var top = "-" + (height / 2) + "px"; var left = "-" + (width / 2) + "px"; var ratio = getRatio(width, height); setStyles(element, { "position": "absolute", "left": "50%", "top": "50%", "margin": top + " 0 0 " + left, "transform": "scale(" + ratio + ", " + ratio + ")" }); } function getRatio(width, height) { return Math.min( document.body.offsetWidth / width, document.body.offsetHeight / height ); } function setStyles(element, styles) { for (var key in styles) { if (element.style.hasOwnProperty(key)) { element.style[key] = styles[key]; } } } fitToScreen(".welcome-screen");
這段代碼不需要額外的庫。它可以在任何支持 CSS 縮放轉換的瀏覽器中工作。最新的 Chrome 版本就足夠了。只需確保 CSS 選擇器與文檔中的元素匹配即可。
這幾個函數接收一個 CSS 選擇器,並將元素居中並縮放以適應屏幕。如果我們在 for 循環中拋出一個錯誤會發生什麼?我們會看到類似這樣的內容……
我們將該函數列表稱為堆棧跟踪。這就是瀏覽器使用的堆棧內部的樣子。它們將分步處理此代碼……
這就像 PHP 如何使用堆棧來存儲上下文一樣。瀏覽器更進一步,為 DOM 事件和 Ajax 回調等內容提供 WebAPI。在其自然狀態下,JavaScript 與 PHP 一樣異步。也就是說:雖然兩者看起來都可以同時執行許多操作,但它們都是單線程的。它們一次只能做一件事情。
使用瀏覽器 WebAPI(例如 setTimeout 和 addEventListener),我們可以將並行工作卸載到不同的線程。當這些事件發生時,瀏覽器將回調添加到回調隊列中。當堆棧下次為空時,瀏覽器從回調隊列中拾取回調並執行它們。
這個清除堆棧,然後是回調隊列的過程,就是事件循環。
在 JS 中,我們可以運行以下代碼:
function fitToScreen(selector) { var element = document.querySelector(selector); var width = element.offsetWidth; var height = element.offsetHeight; var top = "-" + (height / 2) + "px"; var left = "-" + (width / 2) + "px"; var ratio = getRatio(width, height); setStyles(element, { "position": "absolute", "left": "50%", "top": "50%", "margin": top + " 0 0 " + left, "transform": "scale(" + ratio + ", " + ratio + ")" }); } function getRatio(width, height) { return Math.min( document.body.offsetWidth / width, document.body.offsetHeight / height ); } function setStyles(element, styles) { for (var key in styles) { if (element.style.hasOwnProperty(key)) { element.style[key] = styles[key]; } } } fitToScreen(".welcome-screen");
當我們運行這段代碼時,我們會在控制台中看到 outside the timeout,然後是 inside the timeout。 setTimeout 函數是瀏覽器提供的 WebAPI 的一部分。當經過 1 毫秒後,它們會將回調添加到回調隊列中。
第二個 console.log 在來自 inside the setTimeout 的 console.log 開始之前完成。我們在標準 PHP 中沒有類似 setTimeout 的東西,但如果我們必須嘗試模擬它:
setTimeout(function() { console.log("inside the timeout"); }, 1); console.log("outside the timeout");
當我們運行它時,我們會看到 inside the timeout,然後是 outside the timeout。這是因為我們必須在 setTimeout 函數中使用無限循環才能在延遲後執行回調。
將 while 循環移到 setTimeout 之外並將所有代碼包含在其中可能很誘人。這可能會使我們的代碼感覺不那麼阻塞,但在某些時候,我們總是會被該循環阻塞。在某些時候,我們將看到我們一次只能在一個線程中做一件事情。
雖然標準 PHP 中沒有任何類似 setTimeout 的東西,但有一些晦澀的方法可以實現與事件循環並行的非阻塞代碼。我們可以使用 stream_select 等函數來創建非阻塞網絡 IO。我們可以使用像 EIO 這樣的 C 擴展來創建非阻塞文件系統代碼。讓我們來看看基於這些晦澀方法構建的庫……
Icicle 是一個考慮了事件循環的組件庫。讓我們看一個簡單的例子:
function setTimeout(callable $callback, $delay) { $now = microtime(true); while (true) { if (microtime(true) - $now > $delay) { $callback(); return; } } } setTimeout(function() { print "inside the timeout"; }, 1); print "outside the timeout";
這是使用 icicleio/icicle 版本 0.8.0
Icicle 的事件循環實現很棒。它還有許多其他令人印象深刻的功能;例如 A promise、socket 和服務器實現。
Icicle 還使用生成器作為協程。生成器和協程是一個不同的主題,但它們允許的代碼很漂亮:
use Icicle\Loop; Loop\timer(0.1, function() { print "inside timer"; }); print "outside timer"; Loop\run();
這是使用 icicleio/dns 版本 0.5.0 生成器使編寫類似於同步代碼的異步代碼更容易。當與 promise 和事件循環結合使用時,它們會產生像這樣的優秀非阻塞代碼!
ReactPHP 具有類似的事件循環實現,但沒有所有有趣的生成器內容:
function fitToScreen(selector) { var element = document.querySelector(selector); var width = element.offsetWidth; var height = element.offsetHeight; var top = "-" + (height / 2) + "px"; var left = "-" + (width / 2) + "px"; var ratio = getRatio(width, height); setStyles(element, { "position": "absolute", "left": "50%", "top": "50%", "margin": top + " 0 0 " + left, "transform": "scale(" + ratio + ", " + ratio + ")" }); } function getRatio(width, height) { return Math.min( document.body.offsetWidth / width, document.body.offsetHeight / height ); } function setStyles(element, styles) { for (var key in styles) { if (element.style.hasOwnProperty(key)) { element.style[key] = styles[key]; } } } fitToScreen(".welcome-screen");
這是使用 react/event-loop 版本 0.4.1
ReactPHP 比 Icicle 更成熟,並且具有更廣泛的組件。 Icicle 還有很長的路要走才能與 ReactPHP 提供的所有功能競爭。不過,開發人員正在取得良好的進展!
擺脫我們被教導要擁有的單線程思維方式是困難的。如果我們可以訪問非阻塞 API 和事件循環,我們就不知道我們可以編寫多少代碼。
PHP 社區需要了解這種架構。我們需要學習和試驗異步和並行執行。我們需要從擁有事件循環多年的其他語言中竊取這些概念和最佳實踐,直到“如何有效地使用最多的系統資源?”成為一個用 PHP 容易回答的問題。
敬請期待即將推出的 Icicle 的更實際的實現!
PHP 中的事件循環是一種編程結構,用於等待和調度程序中的事件或消息。它通過跟踪響應外部刺激的每個活動事件並在它們完成時調度它們來工作。這在 PHP 中尤其適用於處理異步操作,在這些操作中,您希望啟動操作然後繼續處理而無需等待操作完成。
傳統的 PHP 編程是同步的,這意味著它一次執行一個操作,按照編寫的順序執行,並且必須等待每個操作完成才能繼續執行下一個操作。另一方面,事件循環允許異步編程。這意味著可以啟動操作,然後將其擱置,直到結果準備就緒,在此期間可以執行其他操作。
在您的 PHP 應用程序中實現事件循環涉及使用提供此功能的庫,例如 ReactPHP 或 Amp。這些庫提供了創建和管理事件循環的必要接口和類。然後,您可以使用此事件循環來處理應用程序中的異步任務。
在 PHP 中使用事件循環可以大大提高應用程序的性能和響應能力。它允許您同時處理多個任務,而不是順序處理,這可以導致更好地利用資源並縮短響應時間。這在需要處理大量並發連接的應用程序(例如聊天服務器或實時數據饋送)中尤其有利。
雖然事件循環可以提供顯著的性能優勢,但它們也會使應用程序的複雜性增加。它們需要不同的編程風格,並且可能使代碼更難以理解和調試。此外,並非所有任務都適合異步處理,有些任務在事件循環中可能更難實現。
是的,可以將事件循環與 Laravel 或 Symfony 等 PHP 框架一起使用,儘管這可能需要一些額外的配置。這兩個框架都設計用於與同步 PHP 代碼一起工作,但它們可以適應與事件循環一起工作以處理異步任務。
事件循環中的錯誤處理可能比同步 PHP 代碼中的錯誤處理更複雜。由於任務是異步執行的,因此可能無法立即捕獲錯誤。相反,您通常需要提供一個回調函數,該函數將在發生錯誤時被調用。
是的,您可以在 PHP CLI(命令行界面)腳本中使用事件循環。事實上,這是事件循環的常見用例,因為 CLI 腳本通常需要同時執行多個任務。
PHP 的垃圾收集獨立於事件循環工作。但是,由於事件循環會保留對所有活動任務的引用,因此這些任務只有在完成之後才會被垃圾收集。這意味著您需要注意避免事件循環代碼中的內存洩漏。
是的,您可以將事件循環與 PHP 的內置服務器一起使用。但是,請記住,內置服務器並非設計用於生產用途,並且可能無法提供與專用 Web 服務器相同的性能或可靠性級別。
以上是PHP中的事件循環的介紹的詳細內容。更多資訊請關注PHP中文網其他相關文章!