浅谈Node的内存泄露问题


Posted in NodeJs onMay 06, 2022

1、node内存相关知识

无论是运行在浏览器端的js,还是运行在node中的js,关于内存管理的方案,都是通过垃圾回收机制来实现内存的分配和释放。当我们的代码编写有缺陷时,可能就无法通过gc来释放内存,这个时候,我们就造成了内存泄露。

  • Node.js进程的内存管理,都是由 V8 引擎自动处理的,包括内存的分配和释放。V8 自动处理内存的分配和释放,叫垃圾回收机制(gc)。

  • 垃圾回收机制,解决了大部分内存管理的问题,但是并不意味着内存泄露的问题就不存在。在一些特殊的场景下,会发生内存泄露。

  • 垃圾回收机制,是根据root对象(window/global)依次梳理对象的引用,如果能从root的引用链到达访问,V8就会将其标记为可到达对象,反之为不可到达对象。被标记为不可到达对象(即无引用的对象)后就会被 V8 回收。

2、哪些情况会造成内存泄露

第一、全局变量

全局变量会直接挂在root对象上,不会被清除掉。只要一个变量挂载到root对象上,自程序运行起,它就会一直占据着内存空间。

第二、函数闭包

闭包会引用父级函数的变量,如果闭包未释放,就会导致内存被持续占用,从而导致内存泄露。

在实际的业务场景中,引用的变量极有可能是挂载到从root可以追溯到的对象上,导致的内存泄露。

第三、事件监听

多次的重复的事件监听,可以导致内存泄露的问题。

3、内存泄露的监测

node的内存泄露监测,基本原理如下,

在node中,有专门进行内存监测的工具——heapdump。

heapdump,是一个npm包,使用比较简洁的语法,就能生成内存快照文件。

内存快照文件,是以heapsnapshot为扩展名的一种文件,该文件记录了关于内存使用的基本情况。

Chrome devTools作为一种工具,可以对内存快照文件,进行分析。

一段监测内存变化的代码如下:

const EventEmitter = require('events');
const heapdump = require('heapdump');
global.test = new EventEmitter();
heapdump.writeSnapshot('./' + Date.now() + '.heapsnapshot');
function run3() {
  const innerData = new Buffer(100);
  const outClosure3 = function () {
    void innerData;
  };
  test.on('error', () => {
    console.log('error');
  });
  outClosure3();
}
for(let i = 0; i < 100; i++) {
  run3();
}
// run3();
// gc();
heapdump.writeSnapshot('./' + Date.now() + '.heapsnapshot');

其中,heapdump.writeSnapshot方法,就是生成内存快照文件的方法。

我们使用node的方式,进行执行,就会生成两个heapsnapshot文件。

4、Chrome DevTools进行分析和对比

在Chrome DevTools => Memory => Profiles中,加载生成的两个heapsnapshot文件。

如下图所示,可以查看相应的内存占用情况:

浅谈Node的内存泄露问题

  • Summary:以构造函数名分类显示。
  • Comparison:比较多个快照之间的差异。
  • Containment:查看整个GC路径。
  • Statistics:以饼状图显示内存占用信息。

再看Statistics中关于内存占用的占比分析:

浅谈Node的内存泄露问题

再看Comparison中对比多个快照之间的差异。(可以很明显的看出增加的部分)

浅谈Node的内存泄露问题

5、内存分析的意义

从商业的角度来说,有时候我们无法通过代码,直观的看到内存泄露相关的信息,而这种内存分析的方式,给了我们解决问题的思路。

在现代化的业务开发中,内存监测是一种必要的工具,掌握了基本的原理,有助于我们分析和理解node性能分析的平台型工具。更有助于我们对node的理解。

到此这篇关于浅谈Node的内存泄露的文章就介绍到这了!


Tags in this post...

NodeJs 相关文章推荐
14款NodeJS Web框架推荐
Jul 11 NodeJs
提高NodeJS中SSL服务的性能
Jul 15 NodeJs
使用nodejs中httpProxy代理时候出现404异常的解决方法
Aug 15 NodeJs
NodeJS配置HTTPS服务实例分享
Feb 19 NodeJs
NodeJS测试框架mocha入门教程
Mar 28 NodeJs
NodeJS创建最简单的HTTP服务器
May 15 NodeJs
nodejs之get/post请求的几种方式小结
Jul 26 NodeJs
nodeJS服务器的创建和重新启动的实现方法
May 12 NodeJs
NodeJS 实现多语言的示例代码
Sep 11 NodeJs
nodejs的安装使用与npm的介绍
Sep 11 NodeJs
Nodejs + Websocket 指定发送及群聊的实现
Jan 09 NodeJs
如何利用nodejs实现命令行游戏
Nov 24 NodeJs
分享node.js实现简单登录注册的具体代码
Apr 26 #NodeJs
分享五个Node.js开发的优秀实践 
Apr 07 #NodeJs
Node.js实现爬取网站图片的示例代码
NodeJs使用webpack打包项目的方法详解
Feb 28 #NodeJs
node快速搭建后台的实现步骤
nodejs利用readline提示输入内容实例代码
详解NodeJS模块化
You might like
不错的一篇面向对象的PHP开发模式(简写版)
2007/03/15 PHP
PHP 防注入函数(格式化数据)
2011/08/08 PHP
用C/C++扩展你的PHP 为你的php增加功能
2012/09/06 PHP
php中apc缓存使用示例
2013/12/25 PHP
Json_encode防止汉字转义成unicode的方法
2016/02/25 PHP
PHP常见的序列化与反序列化操作实例分析
2019/10/28 PHP
PHPExcel实现的读取多工作表操作示例
2020/04/14 PHP
PHP 实现 WebSocket 协议原理与应用详解
2020/04/22 PHP
jQuery 1.3 和 Validation 验证插件1.5.1
2009/07/09 Javascript
javascript级联下拉列表实例代码(自写)
2013/05/10 Javascript
jquery中EasyUI实现同步树
2015/03/01 Javascript
自定义百度分享的分享按钮
2015/03/18 Javascript
Javascript中数组方法汇总(推荐)
2015/04/01 Javascript
js+ajax实现获取文件大小的方法
2015/12/08 Javascript
jQuery Masonry瀑布流布局神器使用详解
2017/05/25 jQuery
Bootstrap table中toolbar新增条件查询及refresh参数使用方法
2018/05/18 Javascript
Vue-axios-post数据后端接不到问题解决
2020/01/09 Javascript
vue实现两个组件之间数据共享和修改操作
2020/11/12 Javascript
零基础写python爬虫之打包生成exe文件
2014/11/06 Python
Python中你应该知道的一些内置函数
2017/03/31 Python
python更改已存在excel文件的方法
2018/05/03 Python
Python多进程写入同一文件的方法
2019/01/14 Python
python dict 相同key 合并value的实例
2019/01/21 Python
Python hashlib模块加密过程解析
2019/11/05 Python
Python如何创建装饰器时保留函数元信息
2020/08/07 Python
css3实现信纸/同学录效果的示例代码
2018/12/11 HTML / CSS
CSS3媒体查询Media Queries基础学习教程
2016/02/29 HTML / CSS
Falconeri美国官网:由羊绒和羊毛制成的针织服装
2018/04/08 全球购物
设计部经理的岗位职责
2013/11/16 职场文书
庆七一活动方案
2014/01/25 职场文书
运动会通讯稿500字
2014/02/20 职场文书
自我鉴定书
2014/03/24 职场文书
工程学毕业生自荐信
2014/06/14 职场文书
暑期培训班招生方案
2014/08/26 职场文书
上学路上观后感
2015/06/16 职场文书
Django框架中模型的用法
2022/06/10 Python