# [关于内存泄漏](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服务端