何为V8:
1.V8是 一款主流的JS执行引擎
2.V8采用即时编译 所以速度很快
3.V8有内存设限 32位是不超过800M 64位是不超过1.5G
4.值类型的数据存在栈里由系统回收 一般我们说的回收都是回收堆里的 也就是复杂数据类型的回收
V8 GC 回收策略:
1.V8 采用分代回收的思想
2.内存分为新生代存储区 老生代存储区两种
3.为了区分新生代老生代V8内存空间一分为二 左侧存储新生代 右侧存储老生代
4.针对不同种类对象采用不同的算法以达到高效的处理
V8 常用的GC算法:
1.分代回收 新生代 和 老生代
2.空间复制 From空间复制到To
3.标记清除
4.标记压缩
5.增量标记
新生代对象的垃圾回收
1.小空间用于存储新生代对象 64位最大32M 32位 最大16M
2.新生代对象指的是存活时间较短的对象 例如某函数局部作用域中的一些变量
3.回收过程采用复制算法 + 标记整理
将新生代内存也分为两个大小相同的空间
使用空间为From 空闲空间为To
活动对象存储于From空间中 To一直是空闲的
当From空间应用到 一定程度之后出发GC操作
标记整理后将活动对象复制到To空间中
From空间全部释放 然后与To空间交换
新生代对象回收细节:
1.从From复制到To空间的过程中可能出现晋升(晋升到老生代)
2.晋升: 指的是将新生代对象移动到老生代进行存储
4.一般 一轮GC之后 还存活的新生代对象需要晋升 考虑到这些对象存活时间不短
5.如果To空间的使用率超过25% 那么这些对象也需要晋升
6.因为如果To空间使用率过大 当To变成From之后 有可能没有多少剩余内存留给新进来的数据了
老生代对象的垃圾回收:
1.大空间用于存储老生代对象 64位最大1.4GB 32位最大700M
2.老生代对象指的是存活时间较长的对象例如 全局下的一些变量 或者闭包中的变量
3.回收过程采用 标记清除 标记压缩 增量标记的算法
首先 主要是 使用标记清除完成垃圾空间的回收 标记过得留下 没标记的清除
当有新生代晋升 且老生代内存块不足以存储晋升数据时 触发标记整理 整理碎片化内存
标记压缩 清除对象后会造成堆内存出现碎片的情况 当碎片超过一定限制后UI和启动压缩算法 在压缩过程中 将活的对象像一端移动 直到所有对象都移动完成 然后清理掉不需要的内存
采用增量标记进行效率优化
增量标记 标记过程会阻塞代码执行 所以将整个标记过程分成多次交替的插在执行过程中去标记
采用并发标记 进行效率优化
并发标记 该技术可以让GC 扫描和标记对象的同时 允许JS运行
新生代 老生代垃圾回收的对比:
1.新生代区域的垃圾回收是 使用空间换时间 因为每时每刻都有一半的空间出去空闲状态 通过复制清除的方式快速清理垃圾
2.老生代区域的垃圾会后不适合复制算法 因为老生代 堆内存 数据多 占用内存大 复制浪费时间 浪费内存
老生代中的空间:
enum AllocationSpace {
// TODO(v8:7464): Actually map this space's memory as read-only.
RO_SPACE, // 不变的对象空间
NEW_SPACE, // 新生代用于 GC 复制算法的空间
OLD_SPACE, // 老生代常驻对象空间
CODE_SPACE, // 老生代代码对象空间
MAP_SPACE, // 老生代 map 对象
LO_SPACE, // 老生代大空间对象
NEW_LO_SPACE, // 新生代大空间对象
FIRST_SPACE = RO_SPACE,
LAST_SPACE = NEW_LO_SPACE,
FIRST_GROWABLE_PAGED_SPACE = OLD_SPACE,
LAST_GROWABLE_PAGED_SPACE = MAP_SPACE
};