node 垃圾回收
跟浏览器js不同, 以下代码会找出内存泄露
var theThing = null
var replaceThing = function () {
var originalThing = theThing
var unused = function () {
if (originalThing)
console.log("hi")
}
theThing = {
longStr: new Array(1000000).join('*'),
someMethod: function () {
console.log(someMessage)
}
};
};
setInterval(replaceThing, 1000)
这篇文章中,你可以学到 Node.js 的垃圾回收 (以下简称 GC ) 是怎么工作的,你写下的代码在后台发生了什么,以及内存是如何释放的。
Node.js 应用中的内存管理
每个应用都需要内存才能正常运行。内存管理能动态的分配内存块给需要的程序,在不需要时释放掉,以便能重复使用。
应用级的内存管理可以是手动或自动的。而自动内存管理往往涉及到 GC。
下面的代码片段展示了在 C 中如何使用手动内存管理分配内存:
|
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
int main() {
char name[20];
char *description;
strcpy(name, "RisingStack");
// memory allocation
description = malloc( 30 * sizeof(char) );
if( description == NULL ) {
fprintf(stderr, "Error - unable to allocate required memory\n");
} else {
strcpy( description, "Trace by RisingStack is an APM.");
}
printf("Company name = %s\n", name );
printf("Description: %s\n", description );
// release memory
free(description);
}
|
在手动内存管理中,开发者有责任释放闲置的内存,这种内存管理方式可能会造成下面几个问题:
- 内存泄露,当从不释放使用过的内存时发生
- 野指针,当对象被释放时,而原来的指针仍继续使用。在其他数据覆盖写入或读取敏感信息时会造成严重的安全问题
值得庆幸的是,Node.js 附带了一个垃圾回收器,你不需要去手动管理内存分配
GC 的理念
GC 是一种自动管理应用内存的方法。GC 的工作是回收被未使用的对象所占用的内存。它在 1959 年首次应用于 John McCarthy 创造的 LISP 中。
GC 判断对象不再使用的方式是 没有其他的对象引用它们。 a = o, a = null, o就回收了
GC 前的内存
你的内存看上去如下图所示,如果你有一些互相引用的对象以及一些没有任何引用的对象。
这些没有引用的对象会在 GC 运行 时被回收。
memory-state-before-node-js-garbage-collection
GC 后的内存
当 GC 运行起来,无法访问 (没有引用) 的对象会被删除,同时释放掉相应的内存空间。
memory-state-after-node-js-garbage-collection
GC 的优点
- 防止了野指针 bug
- 不用担心内存的二次释放
- 避免了一些类型的内存泄露
当然,使用 GC 不能解决你所有的问题,而且它也不是内存管理的银弹。
使用 GC 时需要注意的事项
- 性能影响 - GC 会消耗计算能力去决定什么对象应该释放
- 无法预测的停顿 - 现代 GC 实现尝试去避免 stop-the-world 的回收方式
Node.js GC & 内存管理实践
实践出真知,所以我打算通过几段不同的代码向你展示内存中发生了什么
栈
栈上包含了局部变量和 指向堆上对象 或 指 向应用程序控制流程的指针。
在以下示例中,a和b将会被放置在栈中
|
function add (a, b) {
return a + b
}
add(4, 5)
|
堆
堆专门用于存储引用类型对象,如字符串?和对象。
在以下示例中,Car 对象将会被放置在栈中
|
function Car (opts) {
this.name = opts.name
}
const LightningMcQueen = new Car({name: 'Lightning McQueen'})
|
在这之后,内存看起来像这个样子
node-js-garbage-collection-first-step-object-placed-in-memory-heap
让我们添加更多的 Car 对象,看看内存会是什么样子!
|
function Car (opts) {
this.name = opts.name
}
const LightningMcQueen = new Car({name: 'Lightning McQueen'})
const SallyCarrera = new Car({name: 'Sally Carrera'})
const Mater = new Car({name: 'Mater'})
|
node-js-garbage-collection-second-step-more-elements-added-to-the-heap
如果GC现在运行,由于根有对每个对象的引用,没有对象会被释放。
让我们添加一些零件到我们的汽车里 (Car 对象) 使它更有趣一点
|
function Engine (power) {
this.power = power
}
function Car (opts) {
this.name = opts.name
this.engine = new Engine(opts.power)
}
let LightningMcQueen = new Car({name: 'Lightning McQueen', power: 900})
let SallyCarrera = new Car({name: 'Sally Carrera', power: 500})
let Mater = new Car({name: 'Mater', power: 100})
|
node-js-garbage-collection-assigning-values-to-the-objects-in-heap
如果我们不再使用 Mater,但是重新定义并对它赋值 (如Mater = undefined) 会发生什么?
node-js-garbage-collection-redefining-values
结果就是,无法从根上访问 Master 对象。所以当下一次 GC 运行时,它将会被释放:
node-js-garbage-collection-freeing-up-unreachable-object
现在我们了解了 GC 预期行为的基础,那让我们看看它在 V8 中是如何实现的。
GC 方法
在我们之前的一篇文章中,我们讨论了 Node.js GC 方法是如何工作的,所以我强烈建议去阅读这篇文章。
- 新生区和老生区
- 新生代 (Young Generation)
- Scavenge 和 标记删除
一个真实的例子 — The Meteor Case-Study
在 2013 年,Meteor 的作者宣布了他们碰到的关于内存泄露的发现 ?,问题代码如下所示:
var theThing = null
var replaceThing = function () {
var originalThing = theThing
var unused = function () {
if (originalThing)
console.log("hi")
}
theThing = {
longStr: new Array(1000000).join('*'),
someMethod: function () {
console.log(someMessage)
}
};
};
setInterval(replaceThing, 1000)
node 垃圾回收的更多相关文章
- 超详细的node/v8/js垃圾回收机制
前言 垃圾回收器是一把十足的双刃剑.其好处是可以大幅简化程序的内存管理代码,因为内存管理无需程序员来操作,由此也减少了(但没有根除)长时间运转的程序的内存泄漏.对于某些程序员来说,它甚至能够提升代码的 ...
- 深入理解Node.js中的垃圾回收和内存泄漏的捕获
深入理解Node.js中的垃圾回收和内存泄漏的捕获 文章来自:http://wwsun.github.io/posts/understanding-nodejs-gc.html Jan 5, 2016 ...
- Node内存限制与垃圾回收
对象分配 所有的JS对象都是通过堆来进行分配的.使用process.memoryUsage()查看使用情况Node.js 中文网文档 process.memoryUsage() { rss: , he ...
- Java内存模型与垃圾回收
1.Java内存模型 Java虚拟机在执行程序时把它管理的内存分为若干数据区域,这些数据区域分布情况如下图所示: 程序计数器:一块较小内存区域,指向当前所执行的字节码.如果线程正在执行一个Java方法 ...
- ubifs核心功能 -- 垃圾回收
可回收空间的分类 垃圾回收的目的是再利用(回收后的空间大小能写入有效的node),如果再利用的价值越低,其回收的必要性越低.为了进行有效的垃圾回收,UBIFS对可回收空间做了2个层次的水线划分: 死空 ...
- .Net 垃圾回收和大对象处理
CLR垃圾回收器根据所占空间大小划分对象.大对象和小对象的处理方式有很大区别.比如内存碎片整理 —— 在内存中移动大对象的成本是昂贵的,让我们研究一下垃圾回收器是如何处理大对象的,大对象对程序性能有哪 ...
- 浅谈Chrome V8引擎中的垃圾回收机制
垃圾回收器 JavaScript的垃圾回收器 JavaScript使用垃圾回收机制来自动管理内存.垃圾回收是一把双刃剑,其好处是可以大幅简化程序的内存管理代码,降低程序员的负担,减少因 长时间运转而带 ...
- 浅谈V8引擎中的垃圾回收机制
最近在看<深入浅出nodejs>关于V8垃圾回收机制的章节,转自:http://blog.segmentfault.com/skyinlayer/1190000000440270 这篇文章 ...
- 【python进阶】Garbage collection垃圾回收1
前言 GC垃圾回收在python中是很重要的一部分,同样我将分两次去讲解Garbage collection垃圾回收,此篇为Garbage collection垃圾回收第一篇,下面开始今天的说明~~~ ...
随机推荐
- mspdb100.dll不见了的解决办法
一.如果在运行某软件或编译程序时提示缺少.找不到mspdb100.dll等类似提示,将下载来的mspdb100.dll拷贝到指定目录即可 (一般是system系统目录或放到软件同级目录里面),或者重新 ...
- 20145302张薇《Java程序设计》第七周学习总结
20145302 <Java程序设计>第七周学习总结 教材学习内容总结 第十三章 时间的度量 Greenwich Mean Time,格林威治时间,简称GMT时间,由观察太阳而得来: Un ...
- 20145312 《Java程序设计》第三周学习总结
20145312 <Java程序设计>第三周学习总结 学习笔记 Chapter 4 4.1类与对象 4.1.1 定义类 1.以服饰设计为例,类定义用class关键词,建立衣服实例要用new ...
- 20145321 《Java程序设计》第2周学习总结
20145321 <Java程序设计>第2周学习总结 教材学习内容总结 一.类型.变量.运算符 1.类型(基本类型) (1)整数:short(占2字节),int(占4字节),long(占8 ...
- JavaScript的this指针到底指向哪?
编程过程中,着实十分困扰this的指向性,经过查阅一番资料,终于搞清楚了,在这里总结一下,全文分为以下三个部分: 什么是this指针? this指针指向哪里? 何时使用this? 一 什么是this指 ...
- Redis-数据操作
数据操作 redis是key-value的数据,所以每个数据都是一个键值对 键的类型是字符串 值的类型分为五种: 字符串string 哈希hash 列表list 集合set 有序集合zset 数据操作 ...
- redis:Invalid input of type: 'bool' type. Convert to a byte,string or number first
分析:出现此错误的原因是redis版本过高导致的,因此降低redis版本即可 解决: pip install -U redis==2.10.6
- 拓扑排序(dfs)
int c[N];//c[u]=0表示从来没有访问过:c[u]=1表示已经访问过,并且还递归访问过它的所有子:c[u]=-1表示正在访问. int topo[N],t; int G[N][N]; bo ...
- Dijkstra+优先队列
/* Dijkstra的算法思想: 在所有没有访问过的结点中选出dis(s,x)值最小的x 对从x出发的所有边(x,y),更新 dis(s,y)=min(dis(s,y),dis(s,x)+dis(x ...
- ubuntu16.04后续工作
一.ruijie https://blog.csdn.net/u012217085/article/details/24369335 https://blog.csdn.net/Lv_Victor/a ...