溫馨提示×

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

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

swoole平滑重啟的方法

發(fā)布時(shí)間:2020-12-18 10:31:44 來(lái)源:億速云 閱讀:194 作者:小新 欄目:編程語(yǔ)言

這篇文章給大家分享的是有關(guān)swoole平滑重啟的方法的內(nèi)容。小編覺得挺實(shí)用的,因此分享給大家做個(gè)參考。一起跟隨小編過來(lái)看看吧。

swoole 平滑重啟

在swoole中,我們可以向主進(jìn)程發(fā)送各種不同的信號(hào),主進(jìn)程根據(jù)接收到的信號(hào)類型做出不同的處理。比如下面這幾個(gè)

SIGTERM,一種優(yōu)雅的終止信號(hào),會(huì)待進(jìn)程執(zhí)行完當(dāng)前程序之后中斷,而不是直接干掉進(jìn)程

SIGUSR1,將平穩(wěn)的重啟所有的Worker進(jìn)程

SIGUSR2,將平穩(wěn)的重啟所有的Task進(jìn)程

如果我們要實(shí)現(xiàn)重啟server,只需要向主進(jìn)程發(fā)送SIGUSR1信號(hào)就好了。

平滑重啟的原理是當(dāng)主進(jìn)程收到SIGUSR1信號(hào)時(shí),主進(jìn)程就會(huì)向一個(gè)子進(jìn)程發(fā)送安全退出的信號(hào),所謂的安全退出的意思是主進(jìn)程并不會(huì)直接把Worker進(jìn)程殺死,而是等這個(gè)子進(jìn)程處理完手上的工作之后,再讓其光榮的“退休”,最后再拉起新的子進(jìn)程(重新載入新的PHP程序代碼)。

然后再向其他子進(jìn)程發(fā)送“退休”命令,就這樣一個(gè)接一個(gè)的重啟所有的子進(jìn)程。

我們注意到,平滑重啟實(shí)際上就是讓舊的子進(jìn)程逐個(gè)退出并重新創(chuàng)建新的進(jìn)程。為了在平滑重啟時(shí)不影響到用戶,這就要求進(jìn)程中不要保存用戶相關(guān)的狀態(tài)信息,即業(yè)務(wù)進(jìn)程最好是無(wú)狀態(tài)的,避免由于進(jìn)程退出導(dǎo)致信息丟失。

感覺很美好的樣子,凡是重啟只要簡(jiǎn)單的向主進(jìn)程發(fā)送信號(hào)就完事了唄。

理想很豐滿,現(xiàn)實(shí)并非如此。

在swoole中,重啟只能針對(duì)Worker進(jìn)程啟動(dòng)之后載入的文件才有效!什么意思呢,就是說只有在onWorkerStart回調(diào)之后加載的文件,重啟才有意義。在Worker進(jìn)程啟動(dòng)之前就已經(jīng)加載到內(nèi)存中的文件,如果想讓它重新生效,還是只能乖乖的關(guān)閉server再重啟。

說了這么多,我們寫個(gè)例子看看到底怎么樣向主進(jìn)程發(fā)送SIGUSR1信號(hào)以便有效重啟Worker進(jìn)程。

首先我們創(chuàng)建一個(gè)Test類,用于處理onReceive回調(diào)的數(shù)據(jù),為什么要把onReceive回調(diào)的業(yè)務(wù)拿出來(lái)單獨(dú)寫,看完例子你就明白了。

<?php

class Test
{
    public function run($data)
    {
        echo $data;
    }
}

在Test::run方法中,我們第一步僅僅是echo輸出swoole_server接收到的數(shù)據(jù)。

當(dāng)前目錄下我們創(chuàng)建一個(gè)swoole_server的類NoReload.php

<?php
require_once("Test.php");
class NoReload
{
    private $_serv;
    private $_test;
    /**
     * init
     */
    public function __construct()
    {
        $this->_serv = new Swoole\Server("127.0.0.1", 9501);
        $this->_serv->set([
            'worker_num' => 1,
        ]);
        $this->_serv->on('Receive', [$this, 'onReceive']);
        $this->_test = new Test;
    }
    /**
     * start server
     */
    public function start()
    {
        $this->_serv->start();
    }
    public function onReceive($serv, $fd, $fromId, $data)
    {
        $this->_test->run($data);
    }
}
$noReload = new NoReload;
$noReload->start();

特別提醒:我們?cè)诔跏蓟痵woole_server的時(shí)候的寫法是命名空間的寫法

new Swoole\Server

該種風(fēng)格的寫法等同于下劃線寫法 ,swoole對(duì)這兩種風(fēng)格的寫法都支持

new swoole_server

此外我們看下server的代碼邏輯:類定義之前require_once了Test.php,初始化的時(shí)候設(shè)置了一個(gè)Worker進(jìn)程,注冊(cè)了NoReload::onReceive方法為swoole_server的onReceive回調(diào),在onReceive回調(diào)內(nèi)接收到的數(shù)據(jù)傳遞給了Test::run方法處理。

感謝各位的閱讀!關(guān)于swoole平滑重啟的方法就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,讓大家可以學(xué)到更多知識(shí)。如果覺得文章不錯(cuò),可以把它分享出去讓更多的人看到吧!

向AI問一下細(xì)節(jié)

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

AI