# [關于內存泄漏](https://www.workerman.net/doc/webman/others/memory-leak.html#%E5%85%B3%E4%BA%8E%E5%86%85%E5%AD%98%E6%B3%84%E6%BC%8F)
webman是常駐內存框架,所以我們需要稍微關注下內存泄漏的情況。不過開發者不必過于擔心,因為內存泄漏發生在非常極端的條件下,而且很容易規避。webman開發與傳統框架開發體驗基本一致,不必為內存管理做多余的操作。
> **提示**
> webman自帶的monitor進程會監控所有進程內存使用情況,如果進程使用內存即將達到php.ini里`memory_limit`設定的值時,會自動安全重啟對應的進程,達到釋放內存的作用,期間對業務沒有影響。
## [內存泄漏定義](https://www.workerman.net/doc/webman/others/memory-leak.html#%E5%86%85%E5%AD%98%E6%B3%84%E6%BC%8F%E5%AE%9A%E4%B9%89)
隨著請求的不斷增加,webman占用的內存也**無限增加**(注意是**無限增加**),達到幾百M甚至更多,這種是內存泄漏。
如果是內存有增長,但是后面不再增長不算內存泄漏。
一般進程占用幾十M內存是很正常的情況,當進程處理超大請求或者維護海量連接時,單個進程內存占用可能會達到上百M也是常有的事。這部分內存使用后php可能并不會全部交還操作系統。而是留著復用,所以可能會出現處理某個大請求后內存占用變大不釋放內存的情況,這是正常現象。(調用gc\_mem\_caches()方法可以釋放部分空閑內存)
## [內存泄漏是如何發生的](https://www.workerman.net/doc/webman/others/memory-leak.html#%E5%86%85%E5%AD%98%E6%B3%84%E6%BC%8F%E6%98%AF%E5%A6%82%E4%BD%95%E5%8F%91%E7%94%9F%E7%9A%84)
**內存泄漏發生必須滿足以下兩個條件:**
1. 存在**長生命周期的**數組(注意是長生命周期的數組,普通數組沒事)
2. 并且這個**長生命周期的**數組會無限擴張(業務無限向其插入數據,從不清理數據)
如果1 2條件**同時滿足**(注意是同時滿足),那么將會產生內存泄漏。反之不滿足以上條件或者只滿足其中一個條件則不是內存泄漏。
## [長生命周期的數組](https://www.workerman.net/doc/webman/others/memory-leak.html#%E9%95%BF%E7%94%9F%E5%91%BD%E5%91%A8%E6%9C%9F%E7%9A%84%E6%95%B0%E7%BB%84)
webman里長生命周期的數組包括:
1. static關鍵字的數組
2. 單例的數組屬性
3. global關鍵字的數組
> **注意**
> webman中允許使用長生命周期的數據,但是需要保證數據內的數據是有限的,元素個數不會無限擴張。
以下分別舉例說明
#### 無限膨脹的static數組
~~~php
class Foo
{
public static $data = [];
public function index(Request $request)
{
self::$data[] = time();
return response('hello');
}
}
~~~
以`static`關鍵字定義的`$data`數組是長生命周期的數組,并且示例中`$data`數組隨著請求不斷增加而不斷膨脹,導致內存泄漏。
#### 無限膨脹的單例數組屬性
~~~php
class Cache
{
protected static $instance;
public $data = [];
public function instance()
{
if (!self::$instance) {
self::$instance = new self;
}
return self::$instance;
}
public function set($key, $value)
{
$this->data[$key] = $value;
}
}
~~~
調用代碼
~~~php
class Foo
{
public function index(Request $request)
{
Cache::instance()->set(time(), time());
return response('hello');
}
}
~~~
`Cache::instance()`返回一個Cache單例,它是一個長生命周期的類實例,雖然它的`$data`屬性雖然沒有使用`static`關鍵字,但是由于類本身是長生命周期,所以`$data`也是長生命周期的數組。隨著不斷向`$data`數組里添加不同key的數據,程序占用內存也月來越大,造成內存泄漏。
> **注意**
> 如果 Cache::instance()->set(key, value) 添加的key是有限數量的,則不會內存泄漏,因為`$data`數組并沒有無限膨脹。
#### 無限膨脹的global數組
~~~php
class Index
{
public function index(Request $request)
{
global $data;
$data[] = time();
return response($foo->sayHello());
}
}
~~~
global 關鍵字定義的數組并不會在函數或者類方法執行完畢后回收,所以它是長生命周期的數組,以上代碼隨著請求不斷增加會產生內存泄漏。同理在函數或者方法內以static關鍵字定義的數組也是長生命周期的數組,如果數組無限膨脹也會內存泄漏,例如:
~~~php
class Index
{
public function index(Request $request)
{
static $data = [];
$data[] = time();
return response($foo->sayHello());
}
}
~~~
## [建議](https://www.workerman.net/doc/webman/others/memory-leak.html#%E5%BB%BA%E8%AE%AE)
建議開發者不用特別關注內存泄漏,因為它極少發生,如果不幸發生我們可以通過壓測找到哪段代碼產生泄漏,從而定位出問題。即使開發者沒有找到泄漏點,webman自帶的monitor服務會適時安全重啟發生內存泄漏的進程,釋放內存。
如果你實在想盡量規避內存泄漏,可以參考以下建議。
1. 盡量不使用`global`,`static`關鍵字的數組,如果使用確保其不會無限膨脹
2. 對于不熟悉的類,盡量不使用單例,用new關鍵字初始化。如果需要單例,則查看其是否有無限膨脹的數組屬性
- Golang
- Beego框架
- Gin框架
- gin框架介紹
- 使用Gin web框架的知名開源線上項目
- go-admin-gin
- air 熱啟動
- 完整的form表單參數驗證語法
- Go 語言入門練手項目推薦
- Golang是基于多線程模型
- golang 一些概念
- Golang程序開發注意事項
- fatal error: all goroutines are asleep - deadlock
- defer
- Golang 的內建調試器
- go部署
- golang指針重要性
- 包(golang)
- Golang框架選型比較: goframe, beego, iris和gin
- GoFrame
- golang-admin-項目
- go module的使用方法及原理
- go-admin支持多框架的后臺系統(go-admin.cn)
- docker gocv
- go-fac
- MSYS2
- 企業開發框架系統推薦
- gorm
- go-zero
- 優秀系統
- GinSkeleton(gin web 及gin 知識)
- 一次 request -> response 的生命周期概述
- 路由與路由組以及gin源碼學習
- 中間件以及gin源碼學習
- golang項目部署
- 獨立部署golang
- 代理部署golang
- 容器部署golang
- golang交叉編譯
- goravel
- kardianos+gin 項目作為windows服務運行
- go env
- 適用在Windows、Linux和macOS環境下打包Go應用程序的詳細步驟和命令
- Redis
- Dochub
- Docker部署開發go環境
- Docker部署運行go環境
- dochub說明
- Vue
- i18n
- vue3
- vue3基本知識
- element-plus 表格單選
- vue3后臺模板
- Thinkphp
- Casbin權限控制中間件
- 容器、依賴注入、門面、事件、中間件
- tp6問答
- 偽靜態
- thinkphp-queue
- think-throttle
- thinkphp隊列queue的一些使用說明,queue:work和queue:listen的區別
- ThinkPHP6之模型事件的觸發條件
- thinkphp-swoole
- save、update、insert 的區別
- Socket
- workerman
- 介紹
- 從ThinkPHP6移植到Webman的一些技術和經驗(干貨)
- swoole
- swoole介紹
- hyperf
- hf官網
- Swoft
- swoft官網
- easyswoole
- easyswoole官網地址
- EASYSWOOLE 聊天室DEMO
- socket問答
- MySQL
- 聚簇索引與非聚簇索引
- Mysql使用max獲取最大值細節
- 主從復制
- 隨機生成20萬User表的數據
- MySQL進階-----前綴索引、單例與聯合索引
- PHP
- 面向切面編程AOP
- php是單線程的一定程度上也可以看成是“多線程”
- PHP 線程,進程、并發、并行 的理解
- excel數據畫表格圖片
- php第三方包
- monolog/monolog
- league/glide
- 博客-知識網站
- php 常用bc函數
- PHP知識點的應用場景
- AOP(面向切面編程)
- 注解
- 依賴注入
- 事件機制
- phpspreadsheet導出數據和圖片到excel
- Hyperf
- mineAdmin
- 微服務
- nacos注冊服務
- simps-mqtt連接客戶端simps
- Linux
- 切換php版本
- Vim
- Laravel
- RabbitMQ
- thinkphp+rabbitmq
- 博客
- Webman框架
- 框架注意問題
- 關于內存泄漏
- 移動端自動化
- 懶人精靈
- 工具應用
- render
- gitlab Sourcetree
- ssh-agent失敗 錯誤代碼-1
- 資源網站
- Git
- wkhtmltopdf
- MSYS2 介紹
- powershell curl 使用教程
- NSSM(windows服務工具)
- MinGW64
- 知識擴展
- 對象存儲系統
- minio
- 雪花ID
- 請求body參數類型
- GraphQL
- js 深拷貝
- window 共享 centos文件夾
- 前端get/post 請求 特殊符號 “+”傳參數問題
- 什么是SCM系統?SCM系統與ERP系統有什么區別?
- nginx 日志格式統一為 json
- 特殊符號怎么打
- 收藏網址
- 收藏-golang
- 收藏-vue3
- 收藏-php
- 收藏-node
- 收藏-前端
- 規劃ITEM
- 旅游類
- 人臉識別
- dlib
- Docker&&部署
- Docker-compose
- Docker的網絡模式
- rancher
- DHorse
- Elasticsearch
- es與kibana都docke連接
- 4種數據同步到Elasticsearch方案
- GPT
- 推薦系統
- fastposter海報生成
- elasticsearch+logstash+kibana
- beego文檔系統-MinDoc
- jeecg開源平臺
- Java
- 打包部署
- spring boot
- 依賴
- Maven 相關 命令
- Gradle 相關命令
- mybatis
- mybatis.plus
- spring boot 模板引擎
- SpringBoot+Maven多模塊項目(創建、依賴、打包可執行jar包部署測試)完整流程
- Spring Cloud
- Sentinel
- nacos
- Apollo
- java推薦項目
- gradle
- Maven
- Nexus倉庫管理器
- Python
- Masonite框架
- scrapy
- Python2的pip2
- Python3 安裝 pip3
- 安全攻防
- 運維技術
- 騰訊云安全加固建議
- 免費freessl證書申請
- ruby
- homeland
- Protobuf
- GIT
- FFMPEG
- 命令說明
- 音頻
- ffmpeg合并多個MP4視頻
- NODEJS
- 開發npm包
- MongoDB
- php-docker-mongodb環境搭建
- mongo基本命令
- Docker安裝MongoDB最新版并連接
- 少兒編程官網
- UI推薦
- MQTT
- PHP連接mqtt
- EMQX服務端
- php搭建mqtt服務端