浅谈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 win7下安装方法
May 24 NodeJs
使用Nodejs开发微信公众号后台服务实例
Sep 03 NodeJs
NodeJS学习笔记之FS文件模块
Jan 13 NodeJs
nodejs实现HTTPS发起POST请求
Apr 23 NodeJs
使用nodejs开发cli项目实例
Jun 03 NodeJs
nodeJs内存泄漏问题详解
Sep 05 NodeJs
详解nodejs 文本操作模块-fs模块(三)
Dec 22 NodeJs
nodeJS删除文件方法示例
Dec 25 NodeJs
nodejs个人博客开发第六步 数据分页
Apr 12 NodeJs
nodejs爬虫初试superagent和cheerio
Mar 05 NodeJs
NodeJS模块与ES6模块系统语法及注意点详解
Jan 04 NodeJs
详解nodejs内置模块
May 06 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
Linux Apache PHP Oracle 安装配置(具体操作步骤)
2013/06/17 PHP
php解析http获取的json字符串变量总是空白null
2015/03/02 PHP
PHP性能优化大全(php.ini)
2016/05/20 PHP
laravel-admin解决表单select联动时,编辑默认没选上的问题
2019/09/30 PHP
PHP调用QQ互联接口实现QQ登录网站功能示例
2019/10/24 PHP
javascript下操作css的float属性的特殊写法
2007/08/22 Javascript
JavaScript 序列化对象实现代码
2009/12/18 Javascript
Javascript遍历table中的元素示例代码
2014/07/08 Javascript
javascript中clipboardData对象用法详解
2015/05/13 Javascript
优化RequireJS项目的相关技巧总结
2015/07/01 Javascript
AngularJS基础 ng-srcset 指令简单示例
2016/08/03 Javascript
微信小程序tabBar 返回tabBar不刷新页面
2019/07/25 Javascript
Layui 带多选框表格监听事件以及按钮自动点击写法实例
2019/09/02 Javascript
Vue调用后端java接口的实例代码
2019/10/28 Javascript
Websocket 向指定用户发消息的方法
2020/01/09 Javascript
python备份文件的脚本
2008/08/11 Python
python基础知识小结之集合
2015/11/25 Python
python操作MySQL 模拟简单银行转账操作
2017/09/27 Python
python 匹配url中是否存在IP地址的方法
2018/06/04 Python
python3正则提取字符串里的中文实例
2019/01/31 Python
Python中正则表达式的用法总结
2019/02/22 Python
python实现批量文件重命名
2019/10/31 Python
详解如何修改jupyter notebook的默认目录和默认浏览器
2021/01/24 Python
澳大利亚在线购买葡萄酒:The Wine Collective
2020/02/20 全球购物
abstract class和interface有什么区别
2013/08/04 面试题
会计专业应届生求职信
2013/11/24 职场文书
毕业生自我鉴定
2013/12/04 职场文书
应届毕业生如何写求职信
2014/02/16 职场文书
节约能源标语
2014/06/17 职场文书
纪念九一八事变83周年国旗下讲话稿
2014/09/15 职场文书
学校党的群众路线教育实践活动个人整改方案
2014/10/31 职场文书
2015年房地产个人工作总结
2015/05/26 职场文书
七年级作文之秋游
2019/10/21 职场文书
详解Spring事件发布与监听机制
2021/06/30 Java/Android
关于EntityWrapper的in用法
2022/03/22 Java/Android
Mysql 如何合理地统计一个数据库里的所有表的数据量
2022/04/18 MySQL