常驻内存以及如何避免内存泄漏

常驻内存以及如何避免内存泄漏

大家好,又见面了,我是全栈君。

常驻内存以及如何避免内存泄漏

swoole常驻内存

server一开始就把我们的代码加载到内存中了,无论后期我们怎么修改本地磁盘上的代码,客户端再次发起请求的时候,永远都是内存中的代码在生效,所以我们只能终止server,释放内存然后再重启server,重新把新的代码加载到内存中

swoole内存泄漏

因为swoole常驻内存的特性,尤其是server中定义的全局变量(global声明的变量,static静态变量或对象和超全局变量),在使用完之后是不会被释放,久而久之就可能会发生内存溢出。

如何避免内存泄漏

使用max_request 和 task_max_request 来避免内存泄漏
max_request:worker进程的最大任务数,当worker进程处理的任务数超过这个参数时,worker进程会自动退出,以此达到释放内存和资源的目的。

max_request参数使用限制

  1. max_request只能用于同步阻塞、无状态的请求响应式服务器程序

  2. 纯异步的Server不应当设置max_request

  3. 使用Base模式时max_request是无效的

其中Base模式是swoole运行模式的一种,我们主要介绍多进程模式。

总结:

  1. 常驻内存减少了不小开销,swoole不错

  2. 应尽量避免使用全局变量,不用最好,没啥用

  3. max_request可以解决php的内存溢出问题,但是主要还是要养成释放内存的习惯,因为max_request也有限制场景

使用max_request和 task_max_request 可有效避免内存泄漏

server的代码简写

为了方便测试,我们只设置1个Worker进程,1个Task进程,Worker进程的最大任务设置为3次,Task进程的最大任务设置为4次。

$serv = new swoole_server('127.0.0.1', 9501);

$serv->set([
    'worker_num' => 1,
    'task_worker_num' => 1,
    'max_request' => 3,
    'task_max_request' => 4,
]);
$serv->on('Connect', function ($serv, $fd) {
});
$serv->on('Receive', function ($serv, $fd, $fromId, $data) {
    $serv->task($data);
});
$serv->on('Task', function ($serv, $taskId, $fromId, $data) {
});
$serv->on('Finish', function ($serv, $taskId, $data) {
});
$serv->on('Close', function ($serv, $fd) {
});
$serv->start();

client代码

$client = new swoole_client(SWOOLE_SOCK_TCP, SWOOLE_SOCK_SYNC);
$client->connect('127.0.0.1', 9501) || exit("connect failed. Error: {$client->errCode}\n");

// 向服务端发送数据
$client -> send("Just a test.");
$client->close();

客户端第1次请求后,server进程结构

常驻内存以及如何避免内存泄漏

注意进程id为15644和15645,这两个一个是Worker进程,一个是Task进程。

客户端请求第3次后,server进程结构

常驻内存以及如何避免内存泄漏

进程id 15645变成了15680
请求3次后Worker进程自动退出了(并释放内存),然后Manager进程拉起了新的Worker进程(15680)。

客户端请求第4次后,server进程结构

常驻内存以及如何避免内存泄漏

进程id 15644变成了15704
请求第4次后Task进程自动退出了(并释放内存),然后Manager进程拉起了新的Task进程(15704)。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/111882.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • 单片机串口发送数据_单片机烧录找不到串口

    单片机串口发送数据_单片机烧录找不到串口今天用stm32串口转232和工控机通信时,通过笔记本的串口助手与单片机和工控机通信时都很正常,收发都没有问题;但是一用单片机和工控机直接通信就出现通信故障,经过排查发现单片机老是接收到错误数据0X80;当时的波特率是115200,后来查找资料说这是MAX3232ESE-T芯片的极限通信速率了,果断把波特率降低为19200或者其他低于115200的波特率,错误数据就不在出现了;这个问题的很难发…

    2025年11月3日
    4
  • TDSCDMA SIB content[通俗易懂]

    TDSCDMA SIB content[通俗易懂]SIB1:包括NAS系统信息,UE在空闲态和连接态下所使用的定时器和常数信息。 SIB2:URAID信息。 SIB3:小区选择和重选的参数,包括Cellidentity、Cellselectionandre-selectioninfo和CellAccessRestriction三个信息IE。下面对这些IE的内容进行深入剖析。   

    2022年10月4日
    3
  • 深拷贝和浅拷贝的区别,说法正确的是_前端浅拷贝和深拷贝的区别

    深拷贝和浅拷贝的区别,说法正确的是_前端浅拷贝和深拷贝的区别首先,明确一点深拷贝和浅拷贝是针对对象属性为对象的,因为基本数据类型在进行赋值操作时(也就是拷贝)是直接将值赋给了新的变量,也就是该变量是原变量的一个副本,这个时候你修改两者中的任何一个的值都不会影响另一个,而对于对象或者引用数据来说在进行浅拷贝时,只是将对象的引用复制了一份,也就内存地址,即两个不同的变量指向了同一个内存地址,那么在改变任一个变量的值都是该变这个内存地址的所存储的值,所以两个变量的值都会改变。一、clone()方法在Java中是用clone()方法实现深拷贝的,比如以下代码在Jav

    2022年10月1日
    4
  • windows 安装vagrant reload 失败; No Virtualbox Guest Additions installation found.[通俗易懂]

    windows 安装vagrant reload 失败; No Virtualbox Guest Additions installation found.

    2022年2月19日
    51
  • ios防止更新描述文件(ios13屏蔽系统更新描述文件)

    我们以前分享过关于iOS系统屏蔽系统更新提示的方法,但是可能文章中提及的App过于敏感,文章被删除了!刚好有人问iOS系统更新提示怎么去掉?而前段时间描述文件又更新了可以使用了,由于前段时间没有更新,今天我们分享一下详细方法,与以前的通用方法(被删除了的)相比,更加简单!虽说相比简单,但是不同人难易程度的标准可能不同,教程还得详细点,有些步骤会的可以直接跳过!该方法相比的话,比较简单,…

    2022年4月11日
    164
  • 面向过程

    面向过程现在前端技术不是几年前的前端那样,熟练掌握JavaScript、Jquery、Ajax、DOM操作、其他框架等等这些就够了。而随着业务的复杂等多种因素,如今前端有了面向对象的方式编程,如:TypeSc

    2022年8月2日
    7

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号