ホームページ  >  記事  >  バックエンド開発  >  Swooleカスタムプロジェクト初期化イベント処理の実装

Swooleカスタムプロジェクト初期化イベント処理の実装

藏色散人
藏色散人転載
2019-06-03 16:59:112910ブラウズ

最近、Swoole をベースに開発された imi フレームワークを使用してプロジェクトを開発しているときに、プロジェクトの初期化処理を行うという要件に遭遇しました。初期化プロセスが完了する前に Swoole にリクエストを処理させたくない場合があります。読み込まれていない値が存在する可能性があるため、リクエストの処理で問題が発生する可能性が高くなります。

問題を解決するための思考プロセスとデモ コードを以下に示します。

まず、分析後、Swoole はマルチプロセス モードで実行され、MasterManagerWorker プロセスに分割されます。

マスター プロセスは、サービスを開始する cli コマンド ファイルが配置されているプロセスです。ここでの初期化に問題があります。ロードされたすべてのクラスとグローバル変数は、他のワーカー プロセスで使用できますが、他のワーカー プロセスでは使用できません。ホットリスタートが有効になります。

Manager プロセスの状況は基本的に上記と同じです。

するとWorkerプロセスのみが処理を行うことになりますが、WorkerStartイベントに記述しておけば各Workerプロセスが実行することになります。

<strong>WorkerStart</strong> イベント定義:

function onWorkerStart(swoole_server $server, int $worker_id);

$worker_id は 0 ~ $worker_num の数値で、ID がこの Worker プロセスの

であれば、これは簡単に処理でき、workerid が 0 であるかどうかを直接判断して、プロジェクト初期化イベントをトリガーします。残りの問題は、初期化の実行が完了するまですべてのワーカー プロセスがリクエストを処理しないようにする方法です。

考えて試してみた この問題はコルーチンをハングさせることで解決できます デモコードは以下の通りです:

<?php
 
use Swoole\Coroutine;
 
$http = new swoole_http_server(&#39;127.0.0.1&#39;, 8080);
 
$http->on(&#39;WorkerStart&#39;, function(swoole_http_server $server, $workerId){
    $initFlagFile = __DIR__ . &#39;/init.flag&#39;;
    if(0 === $server->worker_id && (!is_file($initFlagFile) || file_get_contents($initFlagFile) != $server->manager_pid))
    {
        // 处理项目初始化事件
        initApp();
        // 写入文件,保证不再重复触发项目初始化事件
        file_put_contents($initFlagFile, $server->manager_pid);
        // 当前worker进程恢复协程
        resumeCos();
        // 通知其它worker进程
        for($i = 1; $i < $server->setting[&#39;worker_num&#39;]; ++$i)
        {
            $server->sendMessage(&#39;init&#39;, $i);
        }
    }
});
 
$http->on(&#39;PipeMessage&#39;, function(swoole_http_server $server, $srcWorkerId, $data) {
    if(0 === $srcWorkerId && &#39;init&#39; === $data && !defined(&#39;APP_INITED&#39;))
    {
        // 其它worker进程恢复协程
        resumeCos();
    }
});
 
$http->on(&#39;request&#39;, function (swoole_http_request $request, swoole_http_response $response) {
    // 判断未初始化完毕,则挂起协程
    if(!defined(&#39;APP_INITED&#39;))
    {
        $GLOBALS[&#39;WORKER_START_END_RESUME_COIDS&#39;][] = Coroutine::getuid();
        Coroutine::suspend();
    }
    $response->header(&#39;content-type&#39;, &#39;text/html;charset=utf-8&#39;);
    $response->end(&#39;IMI 是一款基于 Swoole 开发的协程 PHP 开发框架,拥有常驻内存、协程异步非阻塞IO等优点。官方网站:<a href="https://imiphp.com" target="_blank">https://imiphp.com</a>&#39;);
});
 
$http->start();
 
/**
 * 处理项目初始化事件,比如这里延时5秒,模拟初始化处理
 *
 * @return void
 */
function initApp()
{
    $count = 5;
    for($i = 0; $i < $count; ++$i)
    {
        echo &#39;initing &#39;, ($i + 1), &#39;/&#39;, $count, PHP_EOL;
        sleep(1);
    }
}
 
/**
 * 恢复协程
 *
 * @return void
 */
function resumeCos()
{
    define(&#39;APP_INITED&#39;, true);
    $coids = $GLOBALS[&#39;WORKER_START_END_RESUME_COIDS&#39;] ?? [];
    fwrite(STDOUT, &#39;suspend co count: &#39; . count($coids) . PHP_EOL);
    foreach($coids as $id)
    {
        Coroutine::resume($id);
    }
}

リクエストイベントで初期化が完了したかどうかを判定することで、初期化が完了していない場合はハングします 現在のコルーチンを開始し、コルーチン ID をグローバル変数に追加します。

0 番目のワーカー プロセスが初期化を完了すると、他のワーカー プロセスにメッセージを送信して、中断されていたコルーチンを起動します。初期化期間中に受信したリクエストはこの時点で実行されます。

以上がSwooleカスタムプロジェクト初期化イベント処理の実装の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

声明:
この記事はyurunsoft.comで複製されています。侵害がある場合は、admin@php.cn までご連絡ください。