# 熱重啟
---
[TOC=2,3]
## 1.服務器熱重啟
所謂熱重啟,就是當服務器相關代碼有所變動之后,無需停止服務,而是在服務器仍然運行的狀態下更新文件。Swoole通過內置的reload函數以及兩個自定義信號量實現了這一功能。
Swoole可用的三個信號:SIGTERM,SIGUSR1,SIGUSR2。
* SIGTERM用于停止服務器
* SIGUSR1用于重啟全部的Worker進程
* SIGUSR2用于重啟全部的Task Worker進程。
那要如何實現熱更新代碼文件呢?
Swoole的回調函數中有這個一個回調onWorkerStart,該回調會在Worker進程啟動時被調用。因此,當swoole_server收到SIGUSR1信號并重啟全部Worker進程后,onWorkerStart就會被調用。如果在onWorkerStart中require全部的代碼文件,每次onWorkerStart后都會重新require一次php文件,這樣就能實現代碼文件的熱更新。
* Reload操作只能重新載入Worker進程啟動后加載的PHP文件,建議使用get_included_files函數來列出哪些文件是在WorkerStart之前就加載的PHP文件,在此列表中的PHP文件,即使進行了reload操作也無法重新載入。比如要關閉服務器重新啟動才能生效。
* 對于Server的配置即$serv->set()中傳入的參數設置,必須關閉/重啟整個Server才可以重新加載
來看下代碼實現:
```php
public function onStart( $serv ) {
cli_set_process_title("reload_master");
}
public function onWorkerStart( $serv , $worker_id) {
require_once "reload_page.php";
Test(); // reload_page.php中定義的一個函數
}
```
首先,在onStart回調函數中通過php的cli_set_process_title函數設置進程名。
在onWorkerStart中,require相關的php文件。
然后,新建一個reload.sh文件,輸入如下內容:
```shell
echo "Reloading..."
cmd=$(pidof reload_master)
kill -USR1 "$cmd"
echo "Reloaded"
```
這樣,就可以通過執行這個腳本重啟服務器了。[點此查看完整源碼](https://github.com/LinkedDestiny/swoole-doc/tree/master/src/04/reload)
* SIGUSR1: 向主進程/管理進程發送SIGUSR1信號,將平穩地restart所有worker進程
* 在PHP代碼中可以調用$serv->reload()完成此操作
* swoole的reload有保護機制,當一次reload正在進行時,收到新的重啟信號會丟棄
* 如果設置了user/group,Worker進程可能沒有權限向master進程發送信息,這種情況下必須使用root賬戶,在shell中執行kill指令進行重啟
* reload指令對addProcess添加的用戶進程無效
## 2.停止服務
* SIGTERM: 向主進程/管理進程發送此信號服務器將安全終止
* 在PHP代碼中可以調用$serv->shutdown()完成此操作
## 3.停止task worker
1.7.7版本增加了僅重啟task_worker的功能。只需向服務器發送SIGUSR2即可。
- swoole簡介
- swoole功能概述
- 序章
- 開發必讀
- 1 環境搭建
- 1.1 環境搭建
- 1.2 搭建Echo服務器
- 2 初識Swoole
- 2.1 Worker進程
- 2.2 TaskWorker進程
- 2.3 Timer定時器
- 2.4 Process進程
- 2.5 Table內存表
- 2.6 多端口監聽
- 2.7 sendfile文件支持
- 2.8 SSL支持
- 2.9 熱重啟
- 2.10 http_server
- 附錄*server配置
- 附錄*server函數
- 附錄*server屬性
- 附錄*server回調函數
- 附錄*server高級特性
- 心跳檢測
- 3 Swoole協議
- 3.1 EOF協議
- 3.2 固定包頭協議
- 3.3 Http協議
- 3.4 WebSocket協議
- 3.5 MTQQ協議
- 內置http_server
- 內置websocket_server
- Swoole\Redis\Server
- 4 Swoole異步IO
- 4.1 AsyncIO
- 異步文件系統IO
- swoole_async_readfile
- swoole_async_writefile
- swoole_async_read
- swoole_async_write
- 5 swoole異步客戶端
- ws_client
- http_client
- mysql_client
- redis_client
- tcp_client
- http2_client
- 6 swoole協程
- Swoole\Coroutine\Http\Client
- Swoole\Coroutine\MySQL
- Swoole\Coroutine\Redis
- Coroutine\PostgreSQL
- Swoole\Coroutine\Client
- Swoole\Coroutine\Socket
- Swoole\Coroutine\Channel
- Coroutine
- Swoole\Coroutine::create
- Swoole\Coroutine::resume
- Swoole\Coroutine::suspend
- Swoole\Coroutine::sleep
- Coroutine::getaddrinfo
- Coroutine::gethostbyname
- swoole_async_dns_lookup_coro
- Swoole\Coroutine::getuid
- getDefer
- setDefer
- recv
- Coroutine::stats
- Coroutine::fread
- Coroutine::fget
- Coroutine::fwrite
- Coroutine::readFIle
- Coroutine::writeFIle
- Coroutine::exec
- 7 swoole_process
- process::construct
- process::start
- process::name
- process::signal
- process::setaffinity
- process::exit
- process::kill
- process::daemon
- process->exec
- process::wait
- process::alarm
- 8 swoole定時器
- swoole_timer_tick
- swoole_timer_after
- swoole_timer_clear
- 9 swoole_event
- swoole_event_add
- swoole_event_set
- swoole_event_del
- swoole_event_wait
- swoole_event_defer
- swoole_event_write
- swoole_event_exit
- swoole提供的function
- 常見問題
- 客戶端鏈接失敗原因
- 如何設置進程數
- 如何實現異步任務
- 如何選擇swoole三種模式
- php中哪些函數是阻塞的
- 是否可以共用1個redis或mysql連接
- 如何在回調函數中訪問外部的變量
- 為什么不要send完后立即close
- 不同的Server程序實例間如何通信
- MySQL的連接池、異步、斷線重連
- 在php-fpm或apache中使用swoole
- 學習Swoole需要掌握哪些基礎知識
- 在phpinfo中有在php-m中沒有
- 同步阻塞與異步非阻塞選擇
- CURL發送POST請求服務器端超時
- 附錄
- 預定義常量
- 內核參數調優
- php四種回調寫法
- 守護進程程序常用數據結構
- swoole生命周期
- swoole_server中內存管理機制
- 使用jemalloc優化swoole內存分配性能
- Reactor、Worker、Task的關系
- Manager進程
- Swoole的實現
- Reactor線程
- 安裝擴展
- swoole-worker手冊
- swoole相關開源項目
- 寫在后面的話
- 版本更新記錄
- 4.0