swoole?平滑重啟
在swoole中,我們可以向主進程發送各種不同的信號,主進程根據接收到的信號類型做出不同的處理。比如下面這幾個
SIGTERM,一種優雅的終止信號,會待進程執行完當前程序之后中斷,而不是直接干掉進程? ? ? ? ? (推薦學習: swoole視頻教程)
SIGUSR1,將平穩的重啟所有的Worker進程
SIGUSR2,將平穩的重啟所有的Task進程
如果我們要實現重啟server,只需要向主進程發送SIGUSR1信號就好了。
平滑重啟的原理是當主進程收到SIGUSR1信號時,主進程就會向一個子進程發送安全退出的信號,所謂的安全退出的意思是主進程并不會直接把Worker進程殺死,而是等這個子進程處理完手上的工作之后,再讓其光榮的“退休”,最后再拉起新的子進程(重新載入新的PHP程序代碼)。
然后再向其他子進程發送“退休”命令,就這樣一個接一個的重啟所有的子進程。
我們注意到,平滑重啟實際上就是讓舊的子進程逐個退出并重新創建新的進程。為了在平滑重啟時不影響到用戶,這就要求進程中不要保存用戶相關的狀態信息,即業務進程最好是無狀態的,避免由于進程退出導致信息丟失。
感覺很美好的樣子,凡是重啟只要簡單的向主進程發送信號就完事了唄。
理想很豐滿,現實并非如此。
在swoole中,重啟只能針對Worker進程啟動之后載入的文件才有效!什么意思呢,就是說只有在onWorkerStart回調之后加載的文件,重啟才有意義。在Worker進程啟動之前就已經加載到內存中的文件,如果想讓它重新生效,還是只能乖乖的關閉server再重啟。
說了這么多,我們寫個例子看看到底怎么樣向主進程發送SIGUSR1信號以便有效重啟Worker進程。
首先我們創建一個Test類,用于處理onReceive回調的數據,為什么要把onReceive回調的業務拿出來單獨寫,看完例子你就明白了。
<?php class Test { public function run($data) { echo $data; } }
在Test::run方法中,我們第一步僅僅是echo輸出swoole_server接收到的數據。
當前目錄下我們創建一個swoole_server的類NoReload.php
<?php require_once("Test.php"); class NoReload { private $_serv; private $_test; /** * init */ public function __construct() { $this->_serv?=?new?SwooleServer("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();
特別提醒:我們在初始化swoole_server的時候的寫法是命名空間的寫法
new?SwooleServer
該種風格的寫法等同于下劃線寫法 ,swoole對這兩種風格的寫法都支持
new?swoole_server
此外我們看下server的代碼邏輯:類定義之前require_once了Test.php,初始化的時候設置了一個Worker進程,注冊了NoReload::onReceive方法為swoole_server的onReceive回調,在onReceive回調內接收到的數據傳遞給了Test::run方法處理。