`
hongtoushizi
  • 浏览: 380769 次
  • 性别: Icon_minigender_1
  • 来自: 天津
社区版块
存档分类
最新评论

如何自己检查NodeJS的代码是否存在内存泄漏

阅读更多

原文:http://www.nearform.com/nodecrunch/self-detect-memory-leak-node

追踪NodeJS代码中的内存泄漏一直是一个很有挑战的难题。本文讨论如何从一个node写的应用里自动的跟踪到内存泄漏问题,在这里笔者向大家推荐两款追查内存问题的神器 —— memwatch 和 heapdump

memoryleak

首先,我们来看一个简单的内存泄漏

var http =require('http');var server = http.createServer(function(req, res){for(var i=0; i<1000; i++){
   server.on('request',function leakyfunc(){});}

 res.end('Hello World\n');}).listen(1337,'127.0.0.1');
server.setMaxListeners(0);
console.log('Server running at http://127.0.0.1:1337/. Process PID: ', process.pid);

每一个请求我们增加了1000个导致泄漏的监听器。如果我们在一个shell控制台中执行以下命令:

whiletrue;do curl "http://127.0.0.1:1337/";done

然后在另外一个shell控制台中查看我们的进程

top -pid

我们会看到node进程产生异常高的内存占用,我们的node进程看起来失控了。那么,当我们的node进程出现这种情况的时候,通常我们该怎样诊断出问题的根源?

内存泄露的检测

npm模块 memwatch 是一个非常好的内存泄漏检查工具,让我们先将这个模块安装到我们的app中去,执行以下命令:

npm install --save memwatch

然后,在我们的代码中,添加:

var memwatch =require('memwatch');//memwatch.setup();  原文有这行代码,最新版本的memwatch已去掉这个方法(译者注)

然后监听 leak 事件

memwatch.on('leak',function(info){
 console.error('Memory leak detected: ', info);});

这样当我们执行我们的测试代码,我们会看到下面的信息:

{
 start:FriJan02201510:38:49 GMT+0000(GMT),end:FriJan02201510:38:50 GMT+0000(GMT),
 growth:7620560,
 reason:'heap growth over 5 consecutive GCs (1s) - -2147483648 bytes/hr'}

memwatch发现了内存泄漏!memwatch 判定内存泄漏事件发生的规则如下:

当你的堆内存在5个连续的垃圾回收周期内保持持续增长,那么一个内存泄漏事件被派发

了解更加详细的内容,查看 memwatch

内存泄漏分析

使用memwatch我们发现了存在内存泄漏,这非常好,但是现在呢?我们还需要定位内存泄漏出现的实际位置。要做到这一点,有两种方法可以使用。

memwatch heap diff

通过memwatch你可以得到堆内存使用量和内存随程序运行产生的差异。详细的文档在这里

例如,我们可以在两个leak事件发生的间隔中做一个heap dump

var hd;
memwatch.on('leak',function(info){
 console.error(info);if(!hd){
   hd =new memwatch.HeapDiff();}else{var diff = hd.end();
   console.error(util.inspect(diff,true,null));
   hd =null;}});

执行这段代码会输出更多的信息:

{ before:{
   nodes:244023,
   time:FriJan02201512:13:11 GMT+0000(GMT),
   size_bytes:22095800,
   size:'21.07 mb'},
 after:{
   nodes:280028,
   time:FriJan02201512:13:13 GMT+0000(GMT),
   size_bytes:24689216,
   size:'23.55 mb'},
 change:{
   size_bytes:2593416,
   size:'2.47 mb',
   freed_nodes:388,
   allocated_nodes:36393,
   details:[{ size_bytes:0,'+':0,
   what:'(Relocatable)','-':1,
   size:'0 bytes'},{ size_bytes:0,'+':1,
   what:'Arguments','-':1,
   size:'0 bytes'},{ size_bytes:2856,'+':223,
   what:'Array','-':201,
   size:'2.79 kb'},{ size_bytes:2590272,'+':35987,
   what:'Closure','-':11,
   size:'2.47 mb'},...

所以在内存泄漏事件之间,我们发现堆内存增长了2.47MB,而导致内存增长的罪魁祸首是闭包。如果你的泄漏是由某个class造成的,那么what字段可能会输出具体的class名字,所以这样的话,你会获得足够的信息来帮助你最终定位到泄漏之处。

然而,在我们的例子中,我们唯一获得的信息只是泄漏来自于闭包,这个信息非常有用,但是仍不足以在一个复杂的应用中迅速找到问题的来源(复杂的应用往往有很多的闭包,不知道哪一个造成了内存泄漏——译者注)

所以我们该怎么办呢?这时候该Heapdump出场了。

Heapdump

npm模块node-heapdump是一个非凡的模块,它可以使用来将v8引擎的堆内存内容dump出来,这样你就可以在Chrome的开发者工具中查看问题。你可以在开发工具中对比不同运行阶段的堆内存快照,这样可以帮助你定位到内存泄漏的位置。要想了解heapdump的更多内容,可以阅读这篇文章

现在让我们来试试 heapdump,在每一次发现内存泄漏的时候,我们都将此时的内存堆栈快照写入磁盘中:

memwatch.on('leak',function(info){
 console.error(info);var file ='/tmp/myapp-'+ process.pid +'-'+Date.now()+'.heapsnapshot';
 heapdump.writeSnapshot(file,function(err){if(err) console.error(err);else console.error('Wrote snapshot: '+ file);});});

运行我们的代码,磁盘上会产生一些.heapsnapshot的文件到/tmp目录下。现在,在Chrome浏览器中,启动开发者工具(在mac下的快捷键是alt+cmd+i),点击Profiles标签并点击Load按钮载入我们的快照。

我们能够很清晰地发现原来leakyfunc()是内存泄漏的元凶。

leakyfunc

我们依然还可以通过对比两次记录中heapdump的不同来更加迅速确认两次dump之间的内存泄漏:

heapdump

想要进一步了解开发者工具的memory profiling功能,可以阅读 Taming The Unicorn: Easing JavaScript Memory Profiling In Chrome DevTools 这篇文章。

Turbo Test Runner

我们给Turbo - FeedHenry开发的测试工具提交了一个小补丁 — 使用了上面所说的内存泄漏检查技术。这样就可以让开发者写针对内存的单元测试了,如果模块有内存问题,那么测试结果中就会产生相应的警告。详细了解具体的内容,可以访问Turbo模块。

结论和其他细节

上面的内容讨论了一种检测NodeJS内存泄漏的基本方法,以下是一些结论:

  • heapdump有一些潜规则,例如快照大小等。仔细阅读说明文档,并且生成快照也是比较消耗CPU资源的。
  • 还有些其他方法也能生成快照,各有利弊,针对你的项目选择最适合的方式。(例如,发送sigusr2到进程等等,这里有一个memwatch-sigusr2项目)
  • 需要考虑在什么情况下开启memwatch/heapdump。只有在测试环境中有开启它们的必要,另外也需要考虑heapdump的频度以免耗尽了CPU。总之,选择最适合你项目的方式。
  • 也可以考虑其他的方式来检测内存的增长,比如直接监控process.memoryUsage()是一个可以考虑的方法。
  • 当内存问题被探测到之后,你应该要确定这确实是个内存泄漏问题,然后再告知给相关人员。
  • 当心误判,短暂的内存使用峰值表现得很像是内存泄漏。如果你的app突然要占用大量的CPU和内存,处理时间可能会跨越数个垃圾回收周期,那样的话memwatch很有可能将之误判为内存泄漏。但是,这种情况下,一旦你的app使用完这些资源,内存消耗就会降回正常的水平。所以,你其实需要注意的是持续报告的内存泄漏,而可以忽略一两次突发的警报。
  • memwatch目前仅支持node 0.10.x,node 0.12.x(可能还有io.js)支持的版本在这个分支
  • 这篇文章相关的代码我放在gist上。
  • 转载自: http://www.w3ctech.com/topic/842
分享到:
评论

相关推荐

    nodejs之VM的内存处理测试的demo

    - 检查是否存在内存泄漏,例如,如果在代码执行后内存占用并未下降,可能就需要进一步排查。 通过这样的测试,开发者可以更好地理解Node.js中的内存管理,尤其是VM模块如何影响内存使用,以及如何优化和调试与内存...

    nodeJs内存泄漏问题详解

    Node.js内存泄漏问题详解 内存泄漏在Node.js环境中是一个重要的...通过定期检查内存使用情况,及时优化代码,以及合理配置内存限制,可以有效地防止和管理Node.js中的内存泄漏问题,确保应用程序的高效和稳定运行。

    【JavaScript源代码】NodeJs内存占用过高的排查实战记录.docx

    - **审查代码**:鉴于项目规模较小,开发团队尝试通过代码审查来查找潜在的内存泄漏原因。但这一方法同样未找到明确的答案。 - **聚焦WebSocket连接**:在多次审查快照后,发现与WebSocket相关的关键词频繁出现,...

    nodejs数据.rar

    - 使用适当的垃圾回收策略,避免内存泄漏。 9. **部署与监控**: - PM2:用于Node.js应用的进程管理,支持自动重启、负载均衡等。 - 日志管理:如winston用于记录和分析应用程序日志。 - 性能监控:如New Relic...

    处理pdfmarker中文乱码及内存溢出的处理

    2. **避免循环引用**:检查代码中是否存在循环引用,这可能导致垃圾回收器无法释放内存。使用`WeakRef`对象或及时清理不再需要的对象。 3. **设置最大内存限制**:通过`--max-old-space-size`命令行选项,你可以...

    Nodejs实现自动清理elasticsearch过期索引(日志清理)

    - **资源监控**:清理操作可能涉及大量数据,需关注ES节点的CPU、内存和网络使用情况,避免对生产环境造成影响。 - **异常处理**:编写健壮的代码,处理可能的网络错误、权限问题或其他异常情况。 - **安全性**:...

    Nodejs监控事件循环异常示例详解

    同时,可以进一步分析阻塞的原因,比如是否有内存泄漏,或者是否因为过多的回调导致栈溢出。 总结来说,理解和监控Node.js的事件循环异常对于优化应用性能和避免服务中断至关重要。通过设置合理的阈值和监控机制,...

    一个 npm 包的坎坷“续命”之生

    今天我们文章的主角是 memwatch,一个用来帮助我们检查 Node.js 是否存在内存泄漏的库,和这个库传奇的一生。 2012 年 02 月 06 日,一位 Mozilla 的工程师 lloyd 创建了这个库,并写了一篇博文“Is My NodeJS ...

    Nodejs性能分析优化和分布式设计1

    - `run-valgrind.py`等工具可用于内存泄漏检测。 2. **CPU占用资源分析**: - 利用率的合理范围是用户进程和内核进程分别在65%-75%和30%-35%之间。 - 使用`top`命令分析CPU使用情况,通过`pidstat`跟踪特定进程...

    JavaScript_2024年2月的Nodejs最佳实践列表.zip

    13. **内存管理**:理解V8引擎的垃圾回收机制,避免创建大量短生命周期的对象,可能导致内存泄漏。 14. **性能监控**:使用New Relic、AppDynamics或自定义解决方案来监控应用性能,以便及时发现和解决问题。 15. ...

    ChakraCore由Microsoft实现的一个JavaScript引擎可以嵌入到nodejs中

    3. **内存管理**: ChakraCore有高效的垃圾回收机制,能有效地管理JavaScript对象的生命周期,防止内存泄漏。 4. **多线程支持**: 支持多线程执行JavaScript,这对于处理密集型计算任务或并发操作非常有用。 5. **...

    Node.js-诊断您的Node.js性能问题

    它提供了多种子工具,如`doctor`、`babel`、`profiler`等,分别用于分析CPU使用率、内存泄漏和代码执行情况。通过运行`node-clinic doctor`,您可以得到一份详细的性能报告,包括内存使用、CPU时间、异步调用堆栈等...

    js代码-1.查看运行nodejs 的线上系统的参数

    在实际应用中,根据需求,你可能需要根据特定的业务逻辑处理这些信息,例如检查特定的环境变量是否存在,或者监控内存使用情况以防止内存泄漏。 总结来说,理解和利用Node.js的`process`对象是调试、监控和管理Node...

    iginite_conceitos_nodejs

    9. **性能优化**:Node.js 应用的性能优化通常涉及减少全局变量、避免内存泄漏、正确处理错误、使用高效的算法和数据结构,以及合理利用缓存等。 10. **社区和工具**:Node.js 社区活跃,有许多优秀的库和工具,如 ...

    v12.14.0版本的node.js 6.13.4版本的npm

    此外,Node.js v12.14.0还强化了错误处理,提供了更精确的堆内存泄漏检测,有助于开发者调试和优化代码。 接着,npm(Node Package Manager)是Node.js的包管理器,用于安装、管理和分享第三方模块。npm 6.13.4是6....

    nodejs-webshot

    在处理大量截图任务时,需要考虑如何有效地管理资源,避免内存泄漏或过多的进程占用过多资源。 10. **兼容性和更新**: 保持 Webshot 和 PhantomJS 的版本更新是必要的,因为这两个项目的维护者可能会发布修复和新...

    nodejs-principles:Node.js基础知识存储库

    - 内存管理:合理使用内存,避免内存泄漏,通过`process.memoryUsage()`检查内存使用情况。 7. **其他** - CLI工具开发:Node.js也可用于构建命令行工具,方便开发者日常使用。 - Web框架:Express、Koa、NestJS...

    Node性能跟踪与稳定性优化

    然而,即便如此,在4台服务器上CPU空闲率超过90%,也暗示了可能存在资源未充分利用的情况。 #### 方法与工具 为了更深入地分析和优化Node.js应用的性能,可以采用以下几种方法和工具: 1. **console.time 和 ...

    Nodejs2021

    11. **性能优化**:Node.js 提供了多种工具和最佳实践来优化应用性能,如使用内存分析器找出内存泄漏,使用Benchmark.js进行性能测试,以及利用HTTP缓存策略等。 12. **持续集成/持续部署(CI/CD)**:随着DevOps文化...

    Dangerous:用 nodejs 编写的 Ad Exchange

    然而,Node.js的安全性同样重要,需要正确处理用户输入、防止内存泄漏和确保异步操作的安全。 2. **实时竞价(RTB)系统**:Ad Exchange的核心是RTB,它允许广告商实时竞标广告位,基于用户的兴趣、行为和其他数据...

Global site tag (gtag.js) - Google Analytics