2026年3月14日 星期六

這麼多年後再遇到laravel+swoole,這次我成功建起來了嗎?

想不到洨編主題轉換得這麼快吧,昨天還在那邊前端golang今天就在跟你php laravel + swoole,附上兩份同一間公司給我的JD


GO
1.Go 開發經驗 4 年以上,熟悉 goroutine / channel / context
2.熟悉 Go Web Framework(Gin 優先,Echo、Fiber其次),有 gRPC / protobuf / gRPC-Gateway 經驗加分
3.熟悉 sync package(Mutex / WaitGroup 等)
4.熟悉 Redis / Cache 機制,具 Message Queue 使用經驗(Kafka / RabbitMQ / NATS 任一)
5.具備高併發、高流量場景的系統設計架構經驗
6.加分條件: 有 Laravel 開發及維護經驗
7.加分條件: 有使用過 context/error handling/interface 設計/dependency injection概念,能針對併發控制與效能優化做處理

PHP
1.熟悉 Laravel(實務經驗 2–3 年以上)
2.熟 原生PHP 7 語法與除錯能力
3.具備 Blade / HTML / CSS 基本前端開發修改能力
4.熟 Linux 指令操作(log、排錯、部署)
5.熟 Git(branch/merge/conflict 解決)
6.有 Workerman/Swoole 使用經驗佳

不是7:6的問題,是php我幾乎全部條件都符合,golang我才剛開始,大概只有一半把握,不得不低頭啊

依然是最熟悉的AI學習,沒想到現在出了好多伺服器套件swoole不是唯一選擇,甚至官方都不是預設推薦swoole,是推薦一套叫做RoadRunner,原因是安裝門檻較低。

很久沒碰PHP的我一定不知道差別,所以AI回答如下,各位自行參考

1. 技術特性深度對照

維度Swoole (PHP 擴充)RoadRunner (Go 伺服器)
本質C++ 撰寫的 PHP 擴充功能Go 語言撰寫的 高效能應用伺服器
運作方式修改 PHP 執行行為,直接在 PHP 進程內處理異步、協程。作為獨立的二進制執行檔,透過 RPC (Goridge) 與 PHP 通訊。
安裝門檻較高。需透過 pecl 編譯,常因系統環境(如 Docker、M1/M2 晶片)缺漏共用庫而失敗。極低。只需下載一個編譯好的二進制檔(.exe 或專屬檔案)即可運作。
穩定性效能極致,但若發生 Segment Fault 可能導致整個 PHP 進程崩潰。極高。進程隔離做得更好,即使單個 PHP Worker 壞掉,伺服器主體依然穩固。
功能範疇功能強大,包含協程、定時器、自定義 TCP/UDP 伺服器等。專注於高效能 HTTP 服務,並提供外掛式的功能(如 Queue、KV 儲存)。

2. 我該如何選擇?(推薦指南)

這兩者沒有絕對的高下,只有「適不適合」您的開發環境與團隊背景:

### 推薦選擇 Swoole 的情境:

  • 追求極致效能:如果您需要微秒級的反應速度,Swoole 減少了 RPC 通訊開銷,表現通常更勝一籌。

  • 需要進階異步功能:如果您要在專案中使用 Swoole Table(記憶體共享)、協程(Coroutines) 或底層通訊協定。

  • 已有成熟運維團隊:能搞定 Docker 映像檔編譯與 CI/CD 環境配置的團隊。

### 推薦選擇 RoadRunner 的情境(我的首選建議):

  • 新手或快速開發:安裝過程極度簡單(composer require 後跑個指令就裝好了),不需要處理複雜的編譯問題。

  • 追求系統穩定性:由於它是用 Go 撰寫的獨立伺服器,對 PHP 的侵入性較低,發生致命錯誤(Fatal Error)時的復原機制較為強健。

  • 容器化友善:在 Docker 環境中部署非常輕鬆,不需要為了安裝 PHP 擴充而增加映像檔體積或建構時間。


3. 專家總結

對於大多數的 Laravel 開發者,我更推薦從 RoadRunner 開始

原因在於它的開發者體驗 (DX) 優異,且效能表現已經足以應付 90% 以上的高併發場景。除非您的應用有特殊的協程需求,或是在極限壓力測試下發現 RPC 通訊成為瓶頸,才需要考慮切換到配置較為複雜的 Swoole。

事實上我也沒得選,畢竟是公司自己選,但我也沒想到laravel更新到現在12版,竟然連套件都有了,甚至都寫在官方網站說明頁,照著指令

#因為洨編是mac,所以以下步驟皆為mac的指令,先確保你有php
brew update
brew install php

#安裝composer或更新,可以上官網
php -r "copy('https://getcomposer.org/installer', 'composer-setup.php');"
php -r "if (hash_file('sha384', 'composer-setup.php') === 'c8b085408188070d5f52bcfe4ecfbee5f727afa458b2573b8eaaf77b3419b0bf2768dc67c86944da1544f06fa544fd47') { echo 'Installer verified'.PHP_EOL; } else { echo 'Installer corrupt'.PHP_EOL; unlink('composer-setup.php'); exit(1); }"
php composer-setup.php
php -r "unlink('composer-setup.php');"
sudo mv composer.phar /usr/local/bin/composer

#安裝laravel指令
composer global require laravel/installer

#要把 composer 放到環境變數,你才能使用laravel指令
echo export PATH="$PATH:$HOME/.composer/vendor/bin" >> ~/.zshrc
source ~/.zshrc

#接著透過指令建立新專案
laravel new app

#在新專案中安裝octant
composer require laravel/octane

#手動安裝swoole
prel install swoole

#開啟swoole擴充,php.ini路徑可透過以下路徑尋找
php --ini

#將swoole擴充加入php.ini(路徑請改成自己的)
echo extension="swoole.so" >> /usr/local/etc/php/8.5/php.ini

#檢查swoole是否安裝成功,若畫面上出現 swoole 字樣,代表安裝成功。
php -m | grep swoole

基本上就可以裝起來,是不是看了就覺得很麻煩。再裝不起來可以請AI代勞,洨邊就經歷composer$PATH 沒設定,perl install swoole手動裝裝不好,php.ini沒有開啟swoole套件等細節,官方也沒說,AI幫我找到的,總之在AI幫助下花了點時間還是裝好了,跟當年沒有AI比起來輕鬆不只一點半點。但跟RoadRunner比起來,真的是挺麻煩了,據說只要跑composer就可以弄好!!不管了

然後最讓我驚艷的是,他連前端都整合好了,只要在laravel new app時選擇一下,連前端都幫忙建制起來,甚至還有ssr

最終我們裝好swoole跟octane套件後,把composer.json > scripts > dev & ssr: php artisan start 改成 php artisan octane:start就好,其他文字不要動,就可以直接composer run dev跑起來了(composer run dev......不能說跟前端一模一樣,簡直87%像)

#改完如下,其他都不用動喔,只要改關鍵字
"dev": [
"Composer\\Config::disableProcessTimeout",
"npx concurrently -c \"#93c5fd,#c4b5fd,#fb7185,#fdba74\" \"php artisan octane:start\" \"php artisan queue:listen --tries=1 --timeout=0\" \"php artisan pail --timeout=0\" \"npm run dev\" --names=server,queue,logs,vite --kill-others"
],
#如果你要檔案修改,伺服器會自動重啟需要
#安裝 chokidar:這是 Laravel Octane 監控檔案變化所需的 Node.js 套件。
#更新 composer.json:我已經在 dev 腳本中的 octane:start 指令加入了 --watch 參數
"dev": [
"Composer\\Config::disableProcessTimeout",
"npx concurrently -c \"#93c5fd,#c4b5fd,#fb7185,#fdba74\" \"php artisan octane:start --watch\" \"php artisan queue:listen --tries=1 --timeout=0\" \"php artisan pail --timeout=0\" \"npm run dev\" --names=server,queue,logs,vite --kill-others"
],
# 或是你不一定要ctrl+c,也可以下
php artisan octane:reload



# 正式環境啟動指令
php artisan octane:start --server=swoole --host=0.0.0.0 --port=8000 --workers=auto --max-requests=10000
  • --workers=auto: 根據 CPU 核心數自動分配處理程序。
  • --max-requests=10000: 處理一萬個請求後重啟 Worker,避免 PHP 記憶體洩漏 (Memory Leak)。
# AI是建議用supervisor寫個腳本,因為我是mac,或是你想用PM2應該都可以吧

最終來科普一下有沒有swoole到底有什麼區別

內容有點多,大家想要完整理解,可以直接點這裡,下面是我整理適合PO出來的內容


Q:Workerman/Swoole這兩個東西做什麼用的,然後告訴我跟laravel要如何使用

A:1. Workerman 與 Swoole 是做什麼用的?

傳統的 PHP 運作模式(PHP-FPM)像是一間「吃完就走」的快餐店:

  • 傳統 PHP-FPM: 每個請求進來,PHP 都要重新加載所有檔案、初始化環境,處理完就結束並釋放資源。這在處理 WebSocket(長連接)高頻率請求 時非常低效。

Workerman 與 Swoole 的核心價值: 它們讓 PHP 變成 「常駐記憶體」 的服務。

  • Workerman: 純 PHP 開發的非同步事件驅動框架,相容性高,適合處理 WebSocket、即時通訊。

  • Swoole: 以 C 語言寫成的 PHP 擴展(Extension),效能極高。它提供了協程(Coroutine)能力,能像 Go 語言一樣處理萬級以上的併發。

原本 Laravel 是跑在 PHP-FPM 上,但因為 Laravel 框架啟動(Bootstrapping)時會加載非常多類別,效能會受限。

兩者的結合點: 透過 Workerman 或 Swoole 作為 Laravel 的 「伺服器容器」,讓 Laravel 啟動後常駐在記憶體中。

  • 優點: 請求進來時不需要重新加載核心檔案,效能通常能提升 5~10 倍。

  • 痛點: 必須非常小心「記憶體洩漏(Memory Leak)」和「全域變數污染」,因為變數在請求結束後不會被銷毀。

為何要這樣並存?

  • Swoole: 雖然 Swoole 也能做 WebSocket,但 Laravel Octane 的設計初衷是為了優化 HTTP 效能。若你的專案對 WebSocket 有高度自定義需求(例如特殊的心跳包、TCP 協定解析),Workerman 的純 PHP 環境在除錯與開發上非常靈活。

注意事項:資料庫連接池

  • 痛點: 由於兩者都是「常駐記憶體」,資料庫連線可能會因為逾時而斷開。

  • 解法: 在 Laravel 指令中使用 Workerman 時,需確保資料庫連線有自動重連機制。Laravel 的 DB facade 通常會處理掉大部分情況,但在常駐程序中,最好在 onMessage 裡面執行 DB::reconnect() 以防萬一。

記憶體管理

  • 守則: 嚴禁在這些服務中使用 static 變數來儲存請求相關的資料。

  • 原因: 因為請求結束後變數不會消失,下一個用戶進來會看到上一個用戶的資料(發生污染)。


Q:何謂常駐記憶體
A:

傳統模式 (PHP-FPM):請求結束,煙消雲散

在傳統 Nginx + PHP-FPM 環境下,每一個 HTTP 請求都是獨立的:

  1. 載入: 啟動 PHP 行程,加載 vendor/autoload.php、載入所有 Laravel 核心檔案(Service Providers, Routes...)。

  2. 執行: 處理業務邏輯。

  3. 釋放: 請求結束後,銷毀所有變數、關閉資料庫連線,並清空記憶體

缺點: 每次請求都要重複加載框架,高併發下 CPU 損耗極大。

常駐模式 (Swoole / Workerman):一次啟動,持久運作

PHP 行程啟動後會「常駐」在記憶體中:

  1. 載入: 僅在服務啟動時載入一次框架核心與業務代碼。

  2. 執行: 請求進來時,直接跳到邏輯處理部分,執行完畢後不銷毀行程。

  3. 保持: 等待下一個請求。

優點: 省去了框架初始化的時間,效能提升 5-10 倍。

如果面試官問:「既然這麼快,為什麼不全部都用常駐模式?」你必須提到以下風險:

  1. 記憶體洩漏 (Memory Leak): 如果你在常駐程序中不斷往一個全域數組 (static $data = []) 塞東西卻沒清空,記憶體會被吃光,導致 OOM (Out of Memory)。

  2. 變數污染: 如果沒寫好,A 用戶的登入資訊可能會殘留在全域變數中,被 B 用戶讀取到。

  3. 程式碼熱更新 (Hot Reload): 在 FPM 下,改完代碼存檔即生效;在常駐模式下,你必須重啟服務(或監控檔案更動自動重啟),因為舊的代碼已經編譯並載入記憶體了。


Q:那使用swoole是不是就不用nginx + php-fpm了
A:雖然 Swoole 可以直接對外提供服務,但我們通常會在前面擋一層 Nginx(作為 反向代理 Reverse Proxy),原因如下:
  • 靜態資源處理:Nginx 處理圖片、CSS、JS 等靜態檔案的效能與穩定性遠高於 PHP。讓 Swoole 專心處理動態邏輯(API / Blade)。

  • 負載平衡 (Load Balancing):如果你有多個 Swoole 實例跑在不同伺服器或 Port,Nginx 可以幫你平均分配流量。

  • 安全性 (SSL/TLS):在 Nginx 層級處理 HTTPS 憑證配置非常簡單且成熟,不需要在 PHP 代碼中處理加密。

  • 緩衝區與穩定性:Nginx 可以處理大量慢速連線(Slow Clients),保護後端的 Swoole 不會因為網路波動而被佔滿資源。


Q:在 Laravel 中如何應用協程?
A:
use Laravel\Octane\Facades\Octane;

// 同時執行兩個耗時任務
[$users, $orders] = Octane::concurrently([
    fn () => User::all(),         // 任務 A
    fn () => Order::recent(),     // 任務 B
]);

// 總耗時 = 兩者之中最慢的那個任務,而不是相加。

面試必問:使用協程的禁忌

這部分是面試官用來確認你是否真的「實戰過」的考題:

  1. 不能使用傳統的阻塞函數: 在協程中,嚴禁使用 sleep()curlmysqli 的傳統函數。因為它們會把整個行程鎖死,導致協程失效。

    • 對策:使用 Swoole 提供的協程版本,或開啟一鍵協程化 (Coroutine::set(['hook_flags' => SWOOLE_HOOK_ALL]))。

  2. 全域變數是危險的: 在協程切換過程中,如果你使用 $_GET 或 Laravel 的單例(Singleton)來存個人資料,可能會發生「資料穿透」,讓 A 使用者看到 B 使用者的訂單。

  3. 資料庫連線池 (Connection Pool): 協程並發量很高,如果每個協程都開一個連線,資料庫會瞬間爆掉。

    • 對策:必須配置 連線池,讓協程複用現有的連線。


面試應對總結

當被問到協程時,你的標準回答架構:

  • 定義:協程是輕量級的線程(User-level thread),由開發者在應用層控制切換。

  • 優勢:在處理 I/O 密集型任務(如 API 呼叫、資料庫查詢)時,能極大化利用 CPU。

  • 實踐:在 Laravel 中透過 Octane 與 Swoole 配合,並注意「一鍵協程化」的配置。


Q: 一鍵協程化是什麼,有什麼作用,傳統的阻塞函數有哪些,為什麼會卡住,個別替代方案是什麼? 
A:

1. 什麼是「一鍵協程化」?

在 Swoole 早期,如果你想讓代碼變成非同步,你必須把所有的 PHP 內建函式改寫成 Swoole 專用的 API(例如用 Swoole\Coroutine\Http\Client 取代 curl)。這對既有的 Laravel 專案來說是巨大的災難,因為你不可能改動框架底層。

「一鍵協程化」 (One-click Coroutine Runtime Hook) 的出現解決了這個問題: 它會在底層 劫持 (Hook) 那些傳統的阻塞函數,將它們自動轉化為非同步。

它的作用:

  • 無痛升級:你不需要改動任何 Laravel 的代碼,原本的 DB::query()Http::get() 就會自動具備協程能力。

  • 開發效率:工程師可以繼續沿用 PHP 的同步寫法,但享受非同步的效能。

2. 傳統的阻塞函數有哪些?為什麼會卡住?

為什麼會卡住?

傳統 PHP (FPM) 是 同步阻塞 (Synchronous Blocking) 的。當執行到一個網路請求時,PHP 進程會發出一個系統調用 (System Call),然後 CPU 就會在那裡 乾等 (Wait)

在等待期間,這個進程什麼事都不能做,這就是「卡住」。

常見阻塞函數與替代方案:

類別傳統阻塞函數 (不可直接用)為什麼卡住協程化後的替代 (或自動 Hook 後)
網路請求curl_*, file_get_contents()等待伺服器回應時,整個進程掛起。Swoole\Coroutine\Http\Client 或 Hook 後的 curl
資料庫mysqli_*, PDO::query()等待 SQL 查詢與結果傳回。使用連線池 (Connection Pool) 與 Hook 後的 PDO
快取/儲存Redis::get() (PhpRedis)等待 Redis TCP 回應。Swoole\Coroutine\Redis 或 Hook 後的 Redis
檔案操作file_put_contents(), fopen()等待硬碟讀寫 (Disk I/O)。Swoole\Coroutine\System::readFile
定時/暫停sleep(), usleep()強制讓整個進程休眠。Swoole\Coroutine::sleep() (僅掛起當前協程,不卡進程)

3. 深度對比:sleep() vs Coroutine::sleep()

這是面試最愛考的細節,用來測試你有沒有「協程思維」。

傳統 sleep(2)

  1. 進程執行到這裡。

  2. 核心暫停這個進程 2 秒。

  3. 這 2 秒內,伺服器完全無法接收新的 Request(如果是單進程模式)。

協程 Coroutine::sleep(2)

  1. 協程 A 執行到這裡,發現要等 2 秒。

  2. Swoole 引擎說:「你先去旁邊休息,我把 CPU 讓給協程 B」。

  3. 這 2 秒內,伺服器可以繼續處理幾百個其他人的請求

  4. 2 秒後,引擎再把協程 A 叫醒繼續執行。


4. 面試必殺技:一鍵協程化不是萬能的

當面試官問到這裡,如果你能主動說出限制,你就是全場最強的 Candidate:

「雖然一鍵協程化很強大,但它也有局限性

  1. C 擴展限制:如果某個 PHP 擴展是用 C 寫的,且沒有遵循 PHP 的流 (Stream) 規範,一鍵協程化可能無法劫持它。

  2. 全域變數污染:因為代碼變成了非同步,多個請求可能在同一個進程內交錯執行。如果使用了 $_GETstatic 屬性,會發生資料錯亂。

  3. CPU 密集型任務:如果你的代碼是在做大數運算(而非 I/O 等待),協程也救不了你,因為 CPU 真的在忙,沒辦法切換。」



Q: 資料庫連線池 (Connection Pool):
協程並發量很高,如果每個協程都開一個連線,資料庫會瞬間爆掉。
對策:必須配置 連線池,讓協程複用現有的連線。
這應該怎麼配置,假設我配10個,那意思就是最多十個人同時連線,其他人會塞車嗎?


A:

1. 連線池的運作原理

連線池(Connection Pool)就像是**「公務車車隊」**:

  • 傳統模式:每個人出門都買一台新車,回報後就把車砸掉(開銷大、浪費)。

  • 連線池模式:公司只有 10 台車。有人要出門就去領鑰匙,用完把車開回停車場還給公司,讓下一個人用。

3. 回答你的核心問題:配 10 個會塞車嗎?

答案是:會,但情況跟你想的不一樣。

Q:配 10 個連線,意思就是最多 10 個人同時連線嗎?

是的。 在微秒($μs$)級別的瞬間,確實只有 10 個 SQL 能在資料庫裡執行。

Q:其他人會塞車嗎?

這就是**「協程」「傳統執行緒」**最大的差異:

  1. 排隊機制:當 10 個連線都被借走時,第 11 個請求會進入一個極短暫的等待隊列 (Channel Queue)

  2. 非阻塞等待:因為是協程,第 11 個請求在等待連線時,並不會卡死整個 PHP 進程。它只是「暫停」自己的進度,讓 CPU 先去處理其他不涉及資料庫的邏輯。

  3. 複用率極高:一個 SQL 查詢可能只需要 5ms。這意味著 1 個連線在 1 秒內可以服務 200 個請求。10 個連線在 1 秒內理論上可以處理 2,000 個查詢。

結論:比起「增加連線數」,**「縮短 SQL 執行時間」**對連線池的效能提升更顯著。


4. 面試應對:如何決定連線池的大小?

面試官可能會追問:「那我要怎麼知道該配多少個連線?」你可以這樣回答:

  1. 資料庫承受力:觀察 MySQL 的 max_connections 設定。如果資料庫上限是 100,你 5 個 PHP 節點各配 20 個就滿了。

  2. 壓測觀察

    • 如果連線數太少:請求回應時間(Latency)會變長,因為大家都在排隊等連線。

    • 如果連線數太多:資料庫的 Context Switch 增加,反而導致整體吞吐量下降。

  3. 黃金公式(參考值)

    $$連線數 = (核心數 \times 2) + 硬碟 IO 效能$$

    通常一個 API 服務,連線數設在 20~50 就能應付極大的流量。


5. ⚠️ 注意:連線洩漏 (Connection Leak)

這也是面試必考。如果你的代碼寫了 $pdo = $pool->get(); 但因為報錯(Exception)沒有執行到 $pool->put($pdo);,這個連線就永遠消失在池子裡了。

資深寫法 (Try-Finally):

$pdo = $pool->get();

try {

    // 執行邏輯

} finally {

    $pool->put($pdo); // 確保無論成功或失敗都會歸還連線

}


沒有留言:

張貼留言