## 概述
PHP語言從1995年發布,至今已經有20多年的歷史,在期間涌現了成千上萬的MVC框架,我們大致可以將其分為以下三大類:
**公司內部自研**
有大量的公司會自研MVC框架,會根據自身業務的特性打造適合自身的PHP框架;
**PHP開源框架**
現PHP開源社區流行的PHP MVC框架有Yii、Laravel、CodeIgniter、ZendFrameWork、Symfony等;
**C擴展**
純C擴展的PHP MVC框架由于研發難度大,不易修改,學習成本相對較高,故直到近幾年才出現,比如Yaf、Phalcon等。
上述三類的PHP MVC運行的環境為LA(N)MP,而且其中的A(N)是不可或缺的,也就是說他們都需要依賴Web Server來承接用戶的請求,將請求轉發給PHP進程,解析并執行PHP代碼,而這樣的工作模式是所有PHPer根深蒂固的認知,甚至筆者在前幾年聽到有人說`我們可以拋棄php-fpm,nginx
`,心想簡直是無稽之談。
隨著互聯網技術的發展,大量公司的后端技術架構都在往微服務架構變遷,微服務架構要求我們盡可能的將我們的業務拆分到獨立的部署單元,當然微服務框架的好處是很“誘人”的,但是它會帶來大量的成本開銷和性能開銷,如何在微服務架構實踐中節約成本和提升性能是我們不可邁過的溝壑。
### 那在PHP生態中傳統的LA(N)MP能滿足微服務框架的需求嗎?
**答案顯然是不能的**。就目前PHP的fastcgi進程管理器`php-fpm`和`nginx`的配合已經運行得足夠好,但是由于php-fpm本身是同步阻塞進程模型,在請求結束后釋放所有的資源(包括框架初始化創建的一系列對象),導致PHP進程`空轉(創建銷毀創建)` 消耗大量的CPU資源,從而導致單機的吞吐能力有限。
### 我們是不是應該切換開發語言?
先簡單的來看換開發語言,一個公司或者團隊切換開發語言的成本是巨大的,它直接面臨著放棄已有的技術沉淀,學習新的語言生態和習慣做法,并將現有業務的全面重構,同時整個運維技術棧也要同步更新,需要踩過無數個`“坑”`,所以我們這里暫不展開去講切換開發語言來解決我們的核心問題。
### 那我們還有什么解決方案?
我們分析我們的業務不難發現,90%以上的業務都是IO密集性業務,我們只需要提高IO復用的能力就可以提升單機吞吐能力,另外需要將`php-fpm`同步阻塞模式替換為異步非阻塞模式,當然不一定使用`php-fpm`,就可以解決我們的核心問題——性能。不幸的是目前PHP生態中還沒有一個工程級別的MVC框架能夠滿足我們的需求。
綜上所述,我們需要使用全新的現代化的PHP框架Webman,為微服務架構打下堅實的基礎。
## PHP-FPM工作模式的問題

* nginx基于epoll事件模型,一個worker同時可處理多個請求
* fpm-worker在同一時刻可處理一個請求
* master進程只負責處理worker進程的監控、日志等
* 用戶端請求由elb解析,再經過nginx解析
* fpm-worker每次處理請求前需要重新初始化mvc框架,然后再釋放資源
* 高并發請求時,fpm-worker不夠用,nginx直接響應502
* fpm-worker進程間切換消耗大(如某線上業務在4核8G內存服務器實質可利用16個進程)
## 壓測數據對比
### 目標
1. 對比在相同硬件條件下,PHP Yii2框架在PHP-5.6及PHP7下`echo hello world`的QPS,并觀察平均響應時間及CPU。
2. 對比現有某服務在PHP-5.6及PHP7下的QPS
### 工具
ApacheBench即ab。`ab -n xx -c xx -k http://www.tinywan.com/`
### 硬件
aws c4.xlarge(4核8G)
### Hello World
#### PHP-5.6/Yii2
開啟opcache,64個php-fpm進程,輸出hello world
n | c | qps | 平均響應時間(ms) | CPU |
-------|-------|------------|-----------------|-------|
10000 | 1 | 662.73 | 1.509 | 6% |
50000 | 10 | 1435.35 | 6.967 | 76% |
500000 | 20 | 1577.16 | 13.375 | 83% |
500000 | 50 | 1626.09 | 145.953 | 89% |
#### PHP-7/Yii2
開啟opcache,64個php-fpm進程,輸出hello world
n | c | qps | 平均響應時間(ms) | CPU |
-------|-------|------------|-----------------|-------|
10000 | 1 | 854.81 | 1.170 | 18% |
50000 | 10 | 2008.32 | 4.979 | 62% |
500000 | 20 | 2186.90 | 9.145 | 65% |
500000 | 50 | 2304.98 | 21.692 | 69% |
500000 | 200 | 2331.93 | 85.766 | 69% |
### 某服務
- 1次Redis Get
- 1次 MongoDB Query
- 2個廣告接口
- 2個業務接口
#### PHP-5.6/Yii2
n | c | qps | 平均響應時間(ms) | CPU |
-------|-------|------------|-----------------|-------|
100 | 1 | 4.16 | 240.168 | 9% |
5000 | 5 | 15.36 | 325.502 | 46% |
5000 | 10 | 18.72 | 534.141 | 83% |
5000 | 50 | 19.03 | 2627.159 | 99% |
#### PHP-7/Yii2
n | c | qps | 平均響應時間(ms) | CPU |
-------|-------|------------|-----------------|-------|
100 | 1 | 3.51 | 284.876 | 5% |
5000 | 5 | 17.23 | 290.129 | 21% |
5000 | 10 | 32.36 | 309.057 | 40% |
5000 | 20 | 52.94 | 377.784 | 82% |
5000 | 40 | 55.52 | 720.433 | 91% |
#### 結論
1. PHP-5.6升級PHP-7.0,Yii2框架的性能提升43%
2. PHP-5.6升級PHP-7.0,某服務性能提升170%
3. 單純將業務升級PHP-7.0,能夠提升很大的性能
- 設計模式系列
- 工廠方法模式
- 序言
- Windows程序注冊為服務的工具WinSW
- 基礎
- 安裝
- 開發規范
- 目錄結構
- 配置
- 快速入門
- 架構
- 請求流程
- 架構總覽
- URL訪問
- 容器和依賴注入
- 中間件
- 事件
- 代碼層結構
- 四個層次
- 路由
- 控制器
- 請求
- 響應
- 數據庫
- MySQL實時同步數據到ES解決方案
- 阿里云DTS數據MySQL同步至Elasticsearch實戰
- PHP中的MySQL連接池
- PHP異步非阻塞MySQL客戶端連接池
- 模型
- 視圖
- 注解
- @SpringBootApplication(exclude={DataSourceAutoConfiguration.calss})
- @EnableFeignClients(basePackages = "com.wotu.feign")
- @EnableAspectJAutoProxy
- @EnableDiscoveryClient
- 錯誤和日志
- 異常處理
- 日志處理
- 調試
- 驗證
- 驗證器
- 驗證規則
- 擴展庫
- 附錄
- Spring框架知識體系詳解
- Maven
- Maven和Composer
- 構建Maven項目
- 實操課程
- 01.初識SpringBoot
- 第1章 Java Web發展史與學習Java的方法
- 第2章 環境與常見問題踩坑
- 第3章 springboot的路由與控制器
- 02.Java編程思想深度理論知識
- 第1章 Java編程思想總體
- 第2章 英雄聯盟的小案例理解Java中最為抽象的概念
- 第3章 徹底理解IOC、DI與DIP
- 03.Spring與SpringBoot理論篇
- 第1章 Spring與SpringBoot導學
- 第2章 Spring IOC的核心機制:實例化與注入
- 第3章 SpringBoot基本配置原理
- 04.SprinBoot的條件注解與配置
- 第1章 conditonal 條件注解
- 第2章 SpringBoot自動裝配解析
- 05.Java異常深度剖析
- 第1章 Java異常分類剖析與自定義異常
- 第2章 自動配置Url前綴
- 06.參數校驗機制與LomBok工具集的使用
- 第1章 LomBok工具集的使用
- 第2章 參數校驗機制以及自定義校驗
- 07.項目分層設計與JPA技術
- 第1章 項目分層原則與層與層的松耦合原則
- 第2章 數據庫設計、實體關系與查詢方案探討
- 第3章 JPA的關聯關系與規則查詢
- 08.ORM的概念與思維
- 第1章 ORM的概念與思維
- 第2章 Banner等相關業務
- 第3章 再談數據庫設計技巧與VO層對象的技巧
- 09.JPA的多種查詢規則
- 第1章 DozerBeanMapper的使用
- 第2章 詳解SKU的規格設計
- 第3章 通用泛型Converter
- 10.令牌與權限
- 第1章 通用泛型類與java泛型的思考
- 常見問題
- 微服務
- demo
- PHP中Self、Static和parent的區別
- Swoole-Cli
- 為什么要使用現代化PHP框架?
- 公眾號
- 一鍵部署微信公眾號Markdown編輯器(支持適配和主題設計)
- Autodesigner 2.0發布
- Luya 一個現代化PHP開發框架
- PHPZip - 創建、讀取和管理 ZIP 文件的簡單庫
- 吊打Golang的PHP界天花板webman壓測對比
- 簡潔而強大的 YAML 解析庫
- 推薦一個革命性的PHP測試框架:Kahlan
- ServBay下一代Web開發環境
- 基于Websocket和Canvas實現多人協作實時共享白板
- Apipost預執行腳本如何調用外部PHP語言
- 認證和授權的安全令牌 Bearer Token
- Laradock PHP 的 Docker 完整本地開發環境
- 高效接口防抖策略,確保數據安全,避免重復提交的終極解決方案!
- TIOBE 6月榜單:PHP穩步前行,編程語言生態的微妙變化
- Aho-Corasick字符串匹配算法的實現
- Redis鍵空間通知 Keyspace Notification 事件訂閱
- ServBay如何啟用并運行Webman項目
- 使用mpdf實現導出pdf文件功能
- Medoo 輕量級PHP數據庫框架
- 在PHP中編寫和運行單元測試
- 9 PHP運行時基準性能測試
- QR碼生成器在PHP中的源代碼
- 使用Gogs極易搭建的自助Git服務
- Gitea
- webman如何記錄SQL到日志?
- Sentry PHP: 實時監測并處理PHP應用程序中的錯誤
- Swoole v6 Alpha 版本已發布
- Proxypin
- Rust實現的Redis內存數據庫發布
- PHP 8.4.0 Alpha 1 測試版本發布
- 121
- Golang + Vue 開發的開源輕量 Linux 服務器運維管理面板
- 內網穿透 FRP VS Tailscale
- 新一代開源代碼托管平臺Gitea
- 微服務系列
- Nacos云原生配置中心介紹與使用
- 輕量級的開源高性能事件庫libevent
- 國密算法
- 國密算法(商用密碼)
- GmSSL 支持國密SM2/SM3/SM4/SM9/SSL 密碼工具箱
- GmSSL PHP 使用
- 數據庫
- SQLite數據庫的Web管理工具
- 阿里巴巴MySQL數據庫強制規范
- PHP
- PHP安全測試秘密武器 PHPGGC
- 使用declare(strict_types=1)來獲得更健壯的PHP代碼
- PHP中的魔術常量
- OSS 直傳阿里騰訊示例
- PHP源碼編譯安裝APCu擴展實現數據緩存
- BI性能DuckDB數據管理系統
- 為什么別人可以是架構師!而我卻不是?
- 密碼還在用 MD5 加鹽?不如試試 password_hash
- Elasticsearch 在電商領域的應用與實踐
- Cron 定時任務入門
- 如何動態設置定時任務!而不是寫死在Linux Crontab
- Elasticsearch的四種查詢方式,你知道多少?
- Meilisearch vs Elasticsearch
- OpenSearch vs Elasticsearch
- Emlog 輕量級開源博客及建站系統
- 現代化PHP原生協程引擎 PRipple
- 使用Zephir編寫C擴展將PHP源代碼編譯加密
- 如何將PHP源代碼編譯加密,同時保證代碼能正常的運行
- 為什么選擇Zephir給PHP編寫動態擴展庫?
- 使用 PHP + XlsWriter實現百萬級數據導入導出
- Rust編寫PHP擴展
- 阿里云盤開放平臺對接進行文件同步
- 如何構建自己的PHP靜態可執行文件
- IM后端架構
- RESTful設計方法和規范
- PHP編譯器BPC 7.3 發布,成功編譯ThinkPHP8
- 高性能的配置管理擴展 Yaconf
- PHP實現雪花算法庫 Snowflake
- PHP官方現代化核心加密庫Sodium
- pie
- 現代化、精簡、非阻塞PHP標準庫PSL
- PHP泛型和集合
- 手把手教你正確使用 Composer包管理
- JWT雙令牌認證實現無感Token自動續期
- 最先進PHP大模型深度學習庫TransformersPHP
- PHP如何啟用 FFI 擴展
- PHP超集語言PXP
- 低延遲雙向實時事件通信 Socket.IO
- PHP OOP中的繼承和多態
- 強大的現代PHP高級調試工具Kint
- PHP基金會
- 基于webman+vue3高質量中后臺框架SaiAdmin
- 開源免費的定時任務管理系統:Gocron
- 簡單強大OCR工具EasyOCR在PHP中使用
- PHP代碼抽象語法樹工具PHP AST Viewer
- MySQL數據庫管理工具PHPMyAdmin
- Rust編寫的一款高性能多人代碼編輯器Zed
- 超高性能PHP框架Workerman v5.0.0-beta.8 發布
- 高并發系列
- 入門介紹及安裝
- Lua腳本開發 Hello World
- 執行流程與階段詳解
- Nginx Lua API 接口開發
- Lua模塊開發
- OpenResty 高性能的正式原因
- 記一次查找 lua-resty-mysql 庫 insert_id 的 bug
- 包管理工具OPM和LuaRocks使用
- 異步非阻塞HTTP客戶端庫 lua-resty-http
- Nginx 內置綁定變量
- Redis協程網絡庫 lua-resty-redis
- 動態HTML渲染庫 lua-testy-template
- 單獨的
- StackBlitz在線開發環境
- AI
- 基礎概念
- 12312
- 基礎鏡像的坑
- 利用phpy實現 PHP 編寫 Vision Transformer (ViT) 模型
- 語義化版本 2.0.0