# swoole_server的3種運行模式介紹
[TOC=2,3]
## 一、Base模式(SWOOLE_BASE)
這種模式就是傳統的異步非阻塞Server。在Reactor內直接回調PHP的函數。如果回調函數中有阻塞操作會導致Server退化為同步模式。worker_num參數對與BASE模式仍然有效,swoole會啟動多個Reactor進程。
>BASE模式下reactor和worker是同一個角色
BASE模式的優點:
* BASE模式沒有IPC開銷,性能更好
* BASE模式代碼更簡單,不容易出錯
BASE模式的缺點:
* TCP連接是在worker進程中維持的,所以當某個worker進程掛掉時,此worker內的所有連接都將被關閉
* 少量TCP長連接無法利用到所有Worker進程
* TCP連接與Worker是綁定的,長連接應用中某些連接的數據量大,這些連接所在的Worker進程負載會非常高。但某些連接數據量小,所以在Worker進程的負載會非常低,不同的Worker進程無法實現均衡。
BASE模式的適用場景:
如果客戶端連接之間不需要交互,可以使用BASE模式。如Memcache、Http服務器等。
## 二、線程模式(已廢棄)
這個就是多線程Worker模式,Reactor線程來處理網絡事件輪詢,讀取數據。得到的請求交給Worker線程去處理。 Swoole提供了可配置的參數,以實現m/n的參數調整。在這種模式下onReceive可以有適度的阻塞操作。多線程模式比進程模式輕量一些,而且線程之間可以共享堆棧和資源。
訪問共享內存時會有同步問題,需要使用Swoole提供的鎖機制來保護數據。目前已經提供了Mutex、讀寫鎖、文件鎖、信號量、自旋鎖一共5種鎖的實現。
多線程的優點:
* 對象和變量是共享的,可直接進行操作
* 文件描述符是共享的,不同的線程可以對同一個資源直接進行操作
多線程的缺點:
* 操作非局部變量時需要加鎖,編程難度高
* 一個線程發生內存錯誤,整個進程會全部結束
>由于PHP的ZendVM在多線程模式存在內存錯誤,多線程模式在v1.6.0版本后已關閉
## 三、進程模式(SWOOLE_PROCESS)
多進程模式是最復雜的方式,用了大量的進程間通信、進程管理機制。適合業務邏輯非常復雜的場景。Swoole提供了完善的進程管理、內存保護機制。 在業務邏輯非常復雜的情況下,也可以長期穩定運行。
Swoole在Reactor線程中提供了Buffer的功能,可以應對大量慢速連接和逐字節的惡意客戶端。另外也提供了CPU親和設置選項,使程序運行的效率更好。
進程模式的優點:
* 連接與數據請求發送是分離的,不會因為某些連接數據量大某些連接數據量小導致Worker進程不均衡
* Worker進程發送致命錯誤時,連接并不會被切換
進程模式的缺點:
* 存在2次IPC的開銷,master進程與worker進程需要使用UnixSocket進行通信
* 不支持某些高級功能,如sendwait、pause、resume等操作
- 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