摘要:最近使用基于開發(fā)的框架開發(fā)項(xiàng)目,碰到一個需求,就是想要做項(xiàng)目初始化處理。當(dāng)初始化處理完成前,不想讓處理請求。事件定義是一個從之間的數(shù)字,表示這個進(jìn)程的那這個就好辦了,直接判斷為的去觸發(fā)項(xiàng)目初始化事件。
最近使用基于 Swoole 開發(fā)的 imi 框架 開發(fā)項(xiàng)目,碰到一個需求,就是想要做項(xiàng)目初始化處理。當(dāng)初始化處理完成前,不想讓 Swoole 處理請求。因?yàn)榭赡苡幸恍┲禌]有加載進(jìn)來,處理請求極有可能出現(xiàn)問題。
下面給出了思考過程及解決問題的demo代碼。
首先分析了一下,Swoole 是多進(jìn)程模式運(yùn)行的,分為 Master、Manager、Worker 進(jìn)程。
Master 進(jìn)程就是我們啟動服務(wù)的 cli 命令文件所在進(jìn)程,在這里面初始化有一個問題,這里所有加載的類、全局變量,其它 Worker 進(jìn)程里都可以使用,無法熱重啟生效。
Manager 進(jìn)程的情況基本和上面差不多。
那么只有在 Worker 進(jìn)程做處理了,但如果寫在 WorkerStart 事件里,每個 Worker 進(jìn)程都會去執(zhí)行。
WorkerStart 事件定義:
function onWorkerStart(swoole_server $server, int $worker_id);
$worker_id是一個從0-$worker_num之間的數(shù)字,表示這個Worker進(jìn)程的ID
那這個就好辦了,直接判斷workerid為0的去觸發(fā)項(xiàng)目初始化事件。剩下還有一個問題就是,如何在初始化執(zhí)行完成前,讓所有 Worker 進(jìn)程暫時都不處理請求。
思考并嘗試了一下,這個問題可以通過協(xié)程掛起來解決,demo 代碼如下:
on("WorkerStart", function(swoole_http_server $server, $workerId){ $initFlagFile = __DIR__ . "/init.flag"; if(0 === $server->worker_id && (!is_file($initFlagFile) || file_get_contents($initFlagFile) != $server->manager_pid)) { // 處理項(xiàng)目初始化事件 initApp(); // 寫入文件,保證不再重復(fù)觸發(fā)項(xiàng)目初始化事件 file_put_contents($initFlagFile, $server->manager_pid); // 當(dāng)前worker進(jìn)程恢復(fù)協(xié)程 resumeCos(); // 通知其它worker進(jìn)程 for($i = 1; $i < $server->setting["worker_num"]; ++$i) { $server->sendMessage("init", $i); } } }); $http->on("PipeMessage", function(swoole_http_server $server, $srcWorkerId, $data) { if(0 === $srcWorkerId && "init" === $data && !defined("APP_INITED")) { // 其它worker進(jìn)程恢復(fù)協(xié)程 resumeCos(); } }); $http->on("request", function (swoole_http_request $request, swoole_http_response $response) { // 判斷未初始化完畢,則掛起協(xié)程 if(!defined("APP_INITED")) { $GLOBALS["WORKER_START_END_RESUME_COIDS"][] = Coroutine::getuid(); Coroutine::suspend(); } $response->header("content-type", "text/html;charset=utf-8"); $response->end("IMI 是一款基于 Swoole 開發(fā)的協(xié)程 PHP 開發(fā)框架,擁有常駐內(nèi)存、協(xié)程異步非阻塞IO等優(yōu)點(diǎn)。官方網(wǎng)站:https://imiphp.com"); }); $http->start(); /** * 處理項(xiàng)目初始化事件,比如這里延時5秒,模擬初始化處理 * * @return void */ function initApp() { $count = 5; for($i = 0; $i < $count; ++$i) { echo "initing ", ($i + 1), "/", $count, PHP_EOL; sleep(1); } } /** * 恢復(fù)協(xié)程 * * @return void */ function resumeCos() { define("APP_INITED", true); $coids = $GLOBALS["WORKER_START_END_RESUME_COIDS"] ?? []; fwrite(STDOUT, "suspend co count: " . count($coids) . PHP_EOL); foreach($coids as $id) { Coroutine::resume($id); } }
通過在 request 事件中判斷是否初始化完畢,如果沒有初始化完成,則掛起當(dāng)前協(xié)程,將協(xié)程ID加入全局變量。
當(dāng)?shù)?個 worker 進(jìn)程執(zhí)行完初始化后,通過向其他 worker 進(jìn)程發(fā)送消息,喚醒曾經(jīng)掛起的協(xié)程們,在初始化期間進(jìn)來的請求,這時候會被執(zhí)行。
原文地址:https://blog.yurunsoft.com/a/...
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/30870.html
摘要:是一個基于擴(kuò)展實(shí)現(xiàn)的輕量級高性能的常駐內(nèi)存型的和應(yīng)用服務(wù)框架高度封裝了,,服務(wù)器,以及基于實(shí)現(xiàn)可擴(kuò)展的服務(wù),同時支持包方式安裝部署項(xiàng)目?;趯?shí)用,抽象事件處理類,實(shí)現(xiàn)與底層的回調(diào)的解耦,支持同步異步調(diào)用,內(nèi)置等常用組件等。 swoolefy swoolefy是一個基于swoole擴(kuò)展實(shí)現(xiàn)的輕量級高性能的常駐內(nèi)存型的API和Web應(yīng)用服務(wù)框架,高度封裝了http,websocket,ud...
摘要:和服務(wù)關(guān)系最密切的進(jìn)程是中的進(jìn)程組,絕大部分業(yè)務(wù)處理都在該進(jìn)程中進(jìn)行。隨后觸發(fā)一個事件各組件通過該事件進(jìn)行配置文件加載路由注冊。事件每個請求到來時僅僅會觸發(fā)事件。服務(wù)器生命周期和服務(wù)基本一致,詳情參考源碼剖析功能實(shí)現(xiàn) 作者:bromine鏈接:https://www.jianshu.com/p/4c0...來源:簡書著作權(quán)歸作者所有,本文已獲得作者授權(quán)轉(zhuǎn)載,并對原文進(jìn)行了重新的排版。S...
摘要:官網(wǎng)源碼解讀號外號外歡迎大家我們開發(fā)組定了一個就線下聚一次的小目標(biāo)里面的框架算是非常重的了這里的重先不具體到性能層面主要是框架的設(shè)計思想和框架集成的服務(wù)讓框架可以既可以快速解決很多問題又可以輕松擴(kuò)展中的框架有在應(yīng)該無出其右了這次解讀的源碼 官網(wǎng): https://www.swoft.org/源碼解讀: http://naotu.baidu.com/file/8... 號外號外, 歡迎大...
摘要:以上是服務(wù)啟動過程中的主體設(shè)計,其中包括了各種組件的實(shí)例化,如對象池等。 EasySwoole 服務(wù)啟動過程以及主體設(shè)計流程源碼解析 本文主要講解EasySwoole 服務(wù)的啟動過程,會通過源碼片段講解主體的設(shè)計流程 命令啟動 當(dāng)我們通過php easyswoole start啟動EasySwoole 服務(wù)時,命令真正到達(dá)的文件是 easyswoole項(xiàng)目vendoreasyswool...
摘要:基于擴(kuò)展實(shí)現(xiàn)真正的數(shù)據(jù)庫連接池這種方案中,項(xiàng)目占用的連接數(shù)僅僅為。一種是連接暫時不再使用,其占用狀態(tài)解除,可以從使用者手中交回到空閑隊(duì)列中這種我們稱為連接的歸隊(duì)。源碼剖析系列目錄 作者:bromine鏈接:https://www.jianshu.com/p/1a7...來源:簡書著作權(quán)歸作者所有,本文已獲得作者授權(quán)轉(zhuǎn)載,并對原文進(jìn)行了重新的排版。Swoft Github: https:...
閱讀 1621·2023-04-26 02:43
閱讀 3040·2021-11-11 16:54
閱讀 1362·2021-09-23 11:54
閱讀 1181·2021-09-23 11:22
閱讀 2372·2021-08-23 09:45
閱讀 855·2019-08-30 15:54
閱讀 3107·2019-08-30 15:53
閱讀 3197·2019-08-30 15:53