浅谈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 相关文章推荐
Nodejs进程管理模块forever详解
Jun 01 NodeJs
nodejs开发环境配置与使用
Nov 17 NodeJs
Nodejs多站点切换Htpps协议详解及简单实例
Feb 23 NodeJs
nodeJS(express4.x)+vue(vue-cli)构建前后端分离实例(带跨域)
Jul 05 NodeJs
nodejs使用redis作为缓存介质实现的封装缓存类示例
Feb 07 NodeJs
nodejs中密码加密处理操作详解
Mar 20 NodeJs
NodeJS 实现多语言的示例代码
Sep 11 NodeJs
手把手教你如何使用nodejs编写cli命令行
Nov 05 NodeJs
NodeJS开发人员常见五个错误理解
Oct 14 NodeJs
分享node.js实现简单登录注册的具体代码
Apr 26 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
为什么夜间收到的中波电台比白天多
2021/03/01 无线电
PHP4实际应用经验篇(2)
2006/10/09 PHP
支持中文、字母、数字的PHP验证码
2015/05/04 PHP
Javascript 读后台cookie代码
2008/09/15 Javascript
JavaScript 权威指南(第四版) 读书笔记
2009/08/11 Javascript
基于Jquery的$.cookie()实现跨越页面tabs导航实现代码
2011/03/03 Javascript
以WordPress为例讲解jQuery美化页面Title的方法
2016/05/23 Javascript
BootStrap 智能表单实战系列(十)自动完成组件的支持
2016/06/13 Javascript
教你如何在Node.js中使用jQuery
2016/08/28 Javascript
进阶之初探nodeJS
2017/01/24 NodeJs
js实现简单的获取验证码按钮效果
2017/03/03 Javascript
使用Xcache缓存器加速PHP网站的配置方法
2017/04/22 Javascript
ES6学习教程之对象字面量详解
2017/10/09 Javascript
微信小程序websocket聊天室的实现示例代码
2019/02/12 Javascript
Javascript幻灯片播放功能实现过程解析
2020/05/07 Javascript
js实现飞机大战游戏
2020/08/26 Javascript
javascript实现一款好看的秒表计时器
2020/09/05 Javascript
python文件和目录操作函数小结
2014/07/11 Python
selenium2.0中常用的python函数汇总
2019/08/05 Python
opencv python如何实现图像二值化
2020/02/03 Python
PyCharm 专业版安装图文教程
2020/02/20 Python
Python加载数据的5种不同方式(收藏)
2020/11/13 Python
全面解析CSS Media媒体查询使用操作(推荐)
2017/08/15 HTML / CSS
css3 transform导致子元素固定定位变成绝对定位的方法
2020/03/06 HTML / CSS
用html5绘制折线图的实例代码
2016/03/25 HTML / CSS
加拿大购物频道:The Shopping Channel
2016/07/21 全球购物
ORLY官网:美国专业美甲一线品牌
2019/12/11 全球购物
思想专业自荐信范文
2013/12/25 职场文书
领导干部廉政承诺书
2014/03/27 职场文书
解除劳动合同协议书范本
2014/04/14 职场文书
公务员诚信承诺书
2014/05/26 职场文书
中药学专业毕业生推荐信
2014/07/10 职场文书
党章培训心得体会
2014/09/04 职场文书
护士年终考核评语
2014/12/31 职场文书
助学金感谢信
2015/01/20 职场文书
话题作文之生命的旋律
2019/12/17 职场文书