easyswoole - 基于swoole扩展实现的一款高性能php框架

開發者必讀

  • GitHub 喜歡記得點個 star

社區答疑

  • QQ 交流群

    • VIP群 579434607 (本群需要付費599元)
    • EasySwoole 官方一群 633921431(已滿)
    • EasySwoole 官方二群 709134628(已滿)
    • EasySwoole 官方三群 932625047(已滿)
    • EasySwoole 官方四群 779897753(已滿)
    • EasySwoole 官方五群 853946743(已滿)
    • EasySwoole 官方六群 524475224(已滿)
    • EasySwoole 官方七群 1016674948
  • 商業支持:

    • QQ 291323003
    • EMAIL admin@fosuss.com

注意事項

  • 不要在代碼中執行 sleep 以及其他睡眠函數,這樣會導致整個進程阻塞;協程中可以使用 Co::sleep()
  • exit/die 是危險的,會導致 Worker 進程退出;
  • 可通過 register_shutdown_function 來捕獲致命錯誤,在進程異常退出時做一些清理工作;
  • PHP 代碼中如果有異常拋出,必須在回調函數中進行 try/catch 捕獲異常,否則會導致工作進程退出;
  • EasySwoole 不支持 set_exception_handler,必須使用 try/catch 方式處理異常;
  • 在控制器中不能寫共享 RedisMySQL 等網絡服務客戶端連接的邏輯,每次訪問控制器都必須 new 一個連接

類/函數重復定義

  • 新手非常容易犯這個錯誤,由于 EasySwoole 是常駐內存的,所以加載類/函數定義的文件后不會釋放。因此引入類/函數的 php 文件時必須要使用 include_oncerequire_once,否則會發生 cannot redeclare function/class 的致命錯誤。

建議使用 composer 做自動加載

進程隔離與內存管理

進程隔離也是很多新手經常遇到的問題。修改了全局變量的值,為什么不生效?原因就是全局變量在不同的進程,內存空間是隔離的,所以無效。

所以使用 EasySwoole 開發 Server 程序需要了解 進程隔離 問題。

  • 不同的進程中 PHP 變量不是共享,即使是全局變量,在 A 進程內修改了它的值,在 B 進程內是無效的,如果需要在不同的Worker 進程內共享數據,可以用 RedisMySQL文件Swoole\TableAPCushmget 等工具實現 Worker 進程內共享數據

  • 不同進程的文件句柄是隔離的,所以在 A 進程創建的 Socket 連接或打開的文件,在 B 進程內是無效,即使是將它的 fd 發送到 B 進程也是不可用的。(句柄不能進程共享)

  • 進程克隆。在 Server 啟動時,主進程會克隆當前進程狀態,此后開始進程內數據相互獨立,互不影響。有疑問的新手可以先弄懂PHPpcntl 擴展

EasySwoole 中對象的4層生命周期

開發 Swoole 程序與普通 LAMP 下編程有本質區別。在傳統的 Web 編程中,PHP 程序員只需要關注 request 到達,request 結束即可。而在 Swoole 程序中程序員可以操控更大范圍,變量/對象可以有四種生存周期。

變量、對象、資源、require/include 的文件等下面統稱為對象

程序全局期

EasySwoole 框架根目錄的 bootstrap.php 文件和 EasySwooleEvent.php 文件中的 initialize 事件函數中創建好的對象,我們稱之為程序全局生命周期對象。這些變量只要沒有被作用域銷毀,在程序啟動后就會一直存在,直到整個程序結束運行才會銷毀。

有一些服務器程序可能會連續運行數月甚至數年才會關閉/重啟,那么程序全局期的對象在這段時間內會持續駐留在內存中的。程序全局期對象所占用的內存是 Worker 進程間共享的,不會額外占用內存。

例如:

  • EasySwooleEvent.php 文件中的 initialize 事件函數中使用 Di 注入一個對象,那么在程序開始之后,在EasySwoole 的控制器中,或者其他地方都可以通過 Di 直接調用這個對象
  • bootstrap.php 中引入一個文件 test.php,該文件定義了一個靜態變量,那么在 EasySwoole 的控制器,或者其他地方都可以調用這個靜態變量

這部分內存會在寫時分離(COW),在 Worker 進程內對這些對象進行寫操作時,會自動從共享內存中分離,變為進程全局對象。

例如:

  • EasySwooleEvent.php 文件中的 initialize 事件函數中使用 Di 注入一個對象,并在用戶 A 訪問控制器時修改了這個對象的屬性,那么其他用戶訪問控制器的時候,獲取這個對象屬性時,可能是未改變的狀態(因為不同用戶訪問的控制器所在的進程不同,其他進程不會修改到這個變量,所以需要注意這個問題);
  • bootstrap.php 中引入一個文件 test.php,該文件定義了一個靜態變量 $a = 1,用戶 A 訪問控制器時修改了變量 $a = 2,可能在其他用戶訪問時,依然還是 $a = 1 的狀態。

程序全局期 include/require 的代碼,必須在整個程序 shutdown 時才會釋放,reload 無效

進程全局期

Swoole 擁有進程生命周期控制的機制,Worker 進程啟動后創建的對象(onWorkerStart 中創建的對象或者在控制器中創建的對象),在這個子進程存活周期之內,是常駐內存的。

例如:

  • 程序全局生命周期對象被控制器修改之后,該對象會復制一份出來到控制器所屬的進程,這個對象只能被這個進程訪問,其他進程訪問的依舊是全局對象。
  • 給服務注冊 onWorkerStart 事件(在 EasySwooleEvent.php 中的 mainServerCreate 事件中進行注冊 onWorkerStart 事件)時創建的對象,只會在該 Worker 進程才能獲取到。

進程全局對象所占用的內存是在當前子進程內存堆的,并非共享內存。對此對象的修改僅在當前 Worker 進程中有效,進程全局期 include/require 的文件,在 reload 后就會重新加載

會話期

會話期是在 onConnect 后創建,或者在第一次 onReceive 時創建,onClose 時銷毀。一個客戶端連接進入后,創建的對象會常駐內存,直到此客戶端斷開連接才會銷毀。

LAMP 中,一個客戶端瀏覽器訪問多次網站,就可以理解為會話期。但傳統 PHP 程序,并不能感知到。只有單次訪問時使用 session_start,訪問 $_SESSION 全局變量才能得到會話期的一些信息。

Swoole 中會話期的對象直接是常駐內存的,不需要 session_start 之類操作。可以直接訪問對象,并執行對象的方法。

請求期

請求期是指一個完整的請求發來,也就是 onReceive 收到請求開始處理,直到返回結果發送 response。這個周期所創建的對象,會在請求完成后銷毀。

Swoole 中請求期對象與普通 PHP 程序中的對象就是一樣的。請求到來時創建,請求結束后銷毀。

swoole_server 中內存管理機制

swoole_server 啟動后內存管理的底層原理與普通 php-cli 程序一致。具體請參考 Zend VM 內存管理方面的文章。

局部變量

在事件回調函數返回后,所有局部對象和變量會全部回收,不需要 unset 。如果變量是一個資源類型,那么對應的資源也會被 PHP 底層釋放。

function test()
{
    $a = new Object;
    $b = fopen('/data/t.log', 'r+');
    $c = new swoole_client(SWOOLE_SYNC);
    $d = new swoole_client(SWOOLE_SYNC);
    global $e;
    $e['client'] = $d;
}

$a, $b, $c 都是局部變量,當此函數 return 時,這3個變量會立即釋放,對應的內存會立即釋放,打開的 IO 資源文件句柄會立即關閉。 $d 也是局部變量,但是 return 前將它保存到了全局變量 $e,所以不會釋放。當執行 unset($e['client']) 時,并且沒有任何其他 PHP 變量仍然在引用 $d 變量,那么 $d 就會被釋放。

全局變量

PHP 中,有3類全局變量。

  • 使用 global 關鍵詞聲明的變量
  • 使用 static 關鍵詞聲明的類靜態變量、函數靜態變量
  • PHP 的超全局變量,包括 $_GET、$_POST、$GLOBALS

全局變量和對象,類靜態變量,保存在 swoole_server 對象上的變量不會被釋放。需要程序員自行處理這些變量和對象的銷毀工作。

class Test
{
    static $array = array();
    static $string = '';
}

function onReceive($serv, $fd, $reactorId, $data)
{
    Test::$array[] = $fd;
    Test::$string .= $data;
}
  • 在事件回調函數中需要特別注意非局部變量的 array 類型值,某些操作如 TestClass::$array[] = "string" 可能會造成內存泄漏,嚴重時可能發生爆內存,必要時應當注意清理大數組。
  • 在事件回調函數中,非局部變量的字符串進行拼接操作是必須小心內存泄漏,如 TestClass::$string .= $data,可能會有內存泄漏,嚴重時可能發生爆內存。

解決方法

  • 同步阻塞并且請求響應式無狀態的 Server 程序可以設置 max_request,當 Worker進程/Task進程 結束運行時或達到任務上限后進程自動退出。該進程的所有變量/對象/資源均會被釋放回收。
  • 程序內在 onClose 或設置定時器及時使用 unset 清理變量,回收資源

內存管理部分參照了 Swoole 官方文檔。

約定規范

  • 項目中類名稱與類文件(文件夾)命名,均為大駝峰,變量與類方法為小駝峰。
  • HTTP 服務響應中,業務邏輯代碼中 echo $var 并不會將 $var 內容輸出至瀏覽器頁面相應內容中,請調用 Response 實例中的 wirte() 方法實現。
主站蜘蛛池模板: 毛刷_毛刷辊_工业毛刷辊厂家_毛刷加工制造厂【丰汇刷业】 | 珠海市甜菊科技发展有限公司| 生物除臭箱,玻璃钢拱形盖板_厂家_价格-河北润达环保设备有限公司 | 三轴伺服机械手_五轴伺服机械手_注塑机械手_东莞市浩能自动化机械有限公司 | 康拓威技术(深圳)有限公司|Theia镜头代理商|安讯士AXIS摄像机|安讯士监控系统|博世BOSCH监控|博世会议系统|索尼SONY监控|松下PANASONIC监控|三星韩华SAMSUNG监控|霍尼韦尔Honeywell|海康|大华|华为监控|Theia无畸变镜头|AXIS监控|安讯视摄像机 | 惠声电子、广州市惠声电子科技有限公司、VBS、VBS惠声电子、VBS公共广播生产厂家、VBS广播功放生产厂家、VBS会议系统设备批发、VBSIP网络对讲系统厂家、VBS会议系统厂家、VBS智能中控厂家、VBS专业扩声厂家 | 苏州市相城区望亭镇叶凡工艺服饰绣品厂 | 人工草坪_施工围挡_体育场围网-人工草坪厂家菲斯福 | 液位变送器_智能压力变送器_3051差压变送器_单双法兰,投入式,电容式,温度变送器-淮安润中仪表科技有限公司 | 金刚石线切割-电火花穿孔机-中走丝线切割-泰州锐特 | 数字多媒体展厅设计,智慧科技互动企业展馆展厅设计公司-深圳炫之风 | 医药冷库设计建造-食品保鲜冷库安装-物流冷库工程造价-开冉制冷 医盟网-全国首家医疗信息化行业门户网站 | 螺杆式空压机|沈阳螺杆空压机厂家推荐选择沈阳隆瑚机械有限公司 螺带混合机|卧式螺带混合机|双动力混合机-无锡鑫海干燥粉体设备有限公司 | 运动控制器_数控系统_廊坊市九盈数控技术公司「官网」 | 美缝剂_美缝剂加盟_瓷砖美缝剂_美缝剂厂家_填缝剂_领翔美缝剂-【官网】 | 无塔供水设备_无负压供水设备_变频供水设备_净化过滤设备_加油站油罐_S/F双层油罐_开封市东方供水设备有限公司 | 塑料凹版印刷机-干式复合机-分条机-涂布机生产厂家温州华印机械有限公司 | 石家庄团建公司|石家庄拓展训练|石家庄拓展培训|石家庄公司团建|石家庄拓展公司-石家庄启聚团建公司 | 三坐标测量机_海克斯康三坐标_蔡司三坐标_常州三坐标测量服务中心_苏州长南精密技术有限公司 | 太原重卡叔叔运输有限公司-山西太原大件运输、太原物流公司、太原货运物流、太原大件运输、太原货运信息、长治物流公司、长治大件运输、晋城物流公司、晋城大件运输、忻州大件运输、朔州大件运输、阳泉大件运输、大同大件运输、吕梁大件运输、临汾大件运输、运城大件运城 | 潍坊亿宏重工机械有限公司,破碎机,高性能立磨机,颚式破碎机,锤式破碎机反击式破碎机,重锤式破碎机,高性能反击式破碎机,圆锥式破碎机,给料机系列,链板给料机系列,简易给料机系列,振动给料机 | 智能调光模块,调光系统巨川电气-专注智能照明 | 天象文仪办公家具,25年一站式配齐经验厂家-办公家具官网 | 玉米加工机械_玉米加工设备_玉米深加工机械_玉米糁加工设备--滑县鑫丰粮油机械有限公司 | 酸碱废气中和塔-酸雾废气吸收塔-酸雾废气处理塔|首页-广州市佰镀通风设备有限公司 | 上海中泉泵业制造有限公司-柴油机泵,一体化泵站,柴油机消防泵,泵站,预制一体化泵站 | 辽宁闻天香食品有限公司 | 青浦区摄像头安装/青浦区无线网络覆盖/青浦区网络调试公司/青浦区IT外包公司/金山区网络维护公司/金山区防火墙调试公司 | 面粉加工设备-面粉加工机械-面粉机组-面粉磨粉机-面粉机成套设备-河南粮院机械制造有限公司 | 生物质蒸发器_燃气蒸发器_燃气锅炉价格|厂家直销-山东泰锅锅炉设备有限公司 | 康拓威技术(深圳)有限公司|Theia镜头代理商|安讯士AXIS摄像机|安讯士监控系统|博世BOSCH监控|博世会议系统|索尼SONY监控|松下PANASONIC监控|三星韩华SAMSUNG监控|霍尼韦尔Honeywell|海康|大华|华为监控|Theia无畸变镜头|AXIS监控|安讯视摄像机 | 气体检测仪,多功能气体检测仪,四合一检测仪,氯气检测仪,有机挥发气体检测仪,气体报警器-南京诺邦电子科技有限公司 | 珠海拾比佰彩图板股份有限公司 | 太原万通汽车学校[官网]-太原好的汽修培训学校,学新能源汽车技术,学汽修,学汽车检测与维修技术 | 河北安润防腐管业股份有限公司-远程供液管路_远程供液_远程供液系统 | 膜结构车棚厂家-张拉膜景观棚-汽车充电桩停车棚-嘉兴铭邦膜结构公司 | 激光切割加工,佛山激光切割加工,钣金加工,机加工-佛山市华锐达金属制品有限公司 | 上海物流公司_上海冷链运输_空运_电商仓储配送电话-迈泽物流 | 无塔供水设备_无负压供水设备_变频供水设备_净化过滤设备_加油站油罐_S/F双层油罐_开封市东方供水设备有限公司 | 舟舟宝贝少儿培训网-少儿艺术培训-少儿舞蹈培训-少儿英语培训 | 深圳蓝枫印刷_画册印刷_彩页印刷_宣传册印刷_包装盒印刷_彩盒印刷厂_不干胶印刷厂 |