这篇文章主要介绍 V8 的内存管理和垃圾回收知识。(总结来自网易课堂)
V8引擎内存回收机制
为什么我们要关注内存呢?
- 防止页面占用内存过大,引起客户端卡顿,甚至无响应
- Node使用的也是V8,内存对于后端服务的性能至关重要。因为服务的持久性,后端很容易造成内存溢出。
V8内存结构
在 64 位的机器上,默认最大操作的对象大小约为 1.4G,在 32 位的机器上,默认最大操作的对象大小约为 0.7G。
V8 将内存分为两类:新生代内存空间和老生代内存空间,新生代内存空间主要用来存放存活时间较短的对象,老生代内存空间主要用来存放存活时间较长的对象。 64位新生代的空间为64MB,老生代为1400MB。32位下新生代的空间为16MB,老生代为700MB。
新生代的变量,是一个复制的过程,假设有a,b,c三个变量,如果c不使用了,就会把From中的变量a、b复制到to里面,然后把from里面全部清空。下一次,如果b死掉了,再把a从to复制到from,然后把to清空。
老生代的回收,是标记死掉的变量,在垃圾回收的时候,把死掉的变量全部清除掉,然后进行磁盘碎片处理。
标记清除法主要分三步:
- 将所有的指针(变量名)和分配出去的内存打上标记。
- 从栈区开始查找所有可用的变量名,清除这些变量名的标记,并且清除它们指向的内存的标记。
-
标记清除结束后,回收所有仍然带有标记的内存(说明没有一个有效的变量名指向这块内存)。
image.png
新生代如何晋升到老生代。
必须要满足2个条件:
1.变量必须要经历过回收(也就是复制)
2.To空间已经占用了百分之25。
利用node来查看内存使用情况
通过process.memoryUsage()
查看V8引擎内存其中res是V8总内存,heapTotal:是堆总内存,heapUsed:是堆使用内存,external:是额外C++内存。
我们重点关注一下heapUsed
V8引擎是如何处理变量
内存主要是用来储存变量等数据,局部变量当程序执行结束,且没有引用的时候,就会随之消失。而全局变量会始终存货到程序运行结束。
function getme(){
var mem = process.memoryUsage();
var format = function(bytes){
return (bytes/1024/1024).toFixed(2)+'MB';
};
console.log('Process: heapTotal '+format(mem.heapTotal) +
' heapUsed ' + format(mem.heapUsed) +
' rss ' + format(mem.res)
);
}
上面这个函数可以在node环境中输出V8内存的使用情况。
如何注意内存的使用
优化内存的技巧
- 尽量不要定义全局变量
- 全局变量记得销毁掉
有两种方式,第一种是delete,但是delete在严格模式下是禁止的。
第二种,是将变量 = undefined 或者 变量 = null - 用匿名自执行函数变全局为局部
- 尽量避免闭包
防止内存泄漏
- 滥用缓存
- 大内存量操作
网友评论