13 | 垃圾回收:垃圾数据是如何自动回收的?
该思维导图由 AI 生成,仅供参考
不同语言的垃圾回收策略
- 深入了解
- 翻译
- 解释
- 总结
JavaScript中的垃圾数据回收机制是通过副垃圾回收器和主垃圾回收器来实现的。副垃圾回收器主要负责新生区的垃圾回收,采用Scavenge算法将新生代空间对半划分为对象区域和空闲区域,通过复制和角色翻转完成垃圾数据的回收。而主垃圾回收器则负责老生区的垃圾回收,采用标记-清除和标记-整理算法来处理大对象和长生命周期对象的回收。在执行垃圾回收时,由于JavaScript运行在主线程上,会导致全停顿现象,影响应用的性能和响应能力。为了降低全停顿的影响,V8引擎采用增量标记算法来分解垃圾回收任务,使其与JavaScript应用逻辑交替执行,从而减少页面卡顿现象。总的来说,JavaScript中的垃圾数据回收机制涉及多种算法和策略,需要在满足需求的前提下进行权衡,以适应最核心的需求。文章通过深入浅出的方式介绍了JavaScript中垃圾数据的自动回收机制,为读者提供了全面的技术视角和思考。
《浏览器工作原理与实践》,新⼈⾸单¥59
全部留言(92)
- 最新
- 精选
- mfist1. 如何判断内存泄漏的?一般是感官上的长时间运行页面卡顿,猜可能会有内存泄漏。通过DynaTrace(IE)profiles等工具一段时间收集数据,观察对象的使用情况。然后判断是否存在内存泄漏。修改后验证 2. 工作中避免内存泄漏方法:确定不使用的临时变量置为null,当前es6普及场景下少使用闭包也是一种方法。 今日总结 垃圾回收策略一般分为手动回收和自动回收,java python JavaScript等高级预言为了减轻程序员负担和出错概率采用了自动回收策略。JavaScript的原始类型数据和引用数据是分别存储在栈和椎中的,由于栈和堆分配空间大小差异,垃圾回收方式也不一样。栈中分配空间通过ESP的向下移动销毁保存在栈中数据;堆中垃圾回收主要通过副垃圾回收器(新生代)和主垃圾回收器(老生代)负责的,副垃圾回收器采用scavenge算法将区域分为对象区域和空闲区域,通过两个区域的反转让新生代区域无限使用下去。主垃圾回收器采用Mark-Sweep(Mark-Compact Incremental Marking解决不同场景下问题的算法改进)算法进行空间回收的。无论是主副垃圾回收器的策略都是标记-清除-整理三个大的步骤。另外还有新生代的晋升策略(两次未清除的),大对象直接分配在老生代。
作者回复: 总结很好,还可以通过Chrome开发者工具中的Performance来观察。
2019-09-03122 - Hurry使用 chrome 的 Performance 面板,观察内存变化 如何多次垃圾回收后,整体趋势是向上,就存在内部泄漏的可能!
作者回复: 这是一个很好的方法
2019-09-03325 - 奕对于栈中的垃圾回收,是通过移动 ESP 指针来实现的,是不需要通过V8的垃圾回收机制的吗?
作者回复: 是的 栈中的过期数据直接通过esp给抹掉,效率非常高。
2019-09-03324 - 奕对于新生代,副垃圾回收器是怎么进行标记的,文章也就一句话带过了,是和老生代标记算法一样吗?从一组跟元素开始,然后开始遍历的
作者回复: 新生区和老生区标记过程是同一个过程,之后新生代把存活的数据移动到空闲区,老生代把死去的对象加到空闲列表中。
2019-09-03222 - 芒果大道至简,看完了浏览器的垃圾回收,让我联想到了jvm的垃圾回收,发现2者思想上基本都差不多。
作者回复: 现代虚拟机都是抄来抄去的
2019-11-06221 - 屈悦微这篇文章写得很有深度,反复看了几遍,收获颇多,但是仍有一个问题,望作者百忙之中能解答 在本篇中作者介绍了的垃圾回收机制是,标记对象的机制 但在《javascript高级程序设计》中还介绍了引用计数的机制 我产生以下两个疑问 1.v8有没有使用引用计数的机制? 2.如果有,何时使用引用计数,何时使用标记对象?
作者回复: 引用计数有问题,会导致内存泄漏,所以现在流行的垃圾回收器都没有采用引用计数的方式!
2019-12-0617 - 郝仁杰trim之后,数据在堆上的地址发生变化,v8是如何更新对应栈上的引用的
作者回复: JavaScript中的原始字符串是不可变的(immutable),也就是说,一旦一个字符串创建了,它在内存中的值就不可能改变,这和其他语言是有区别的。 所以当你调用trim方法后,v8引擎返回给你的是一个新字符串,并不是之前的字符串了。
2019-09-03716 - 咖飞的白请教老师几个问题: 1. JS 执行代码时是在执行声明语句时就分配内存还是赋值时分配?若是执行声明语句时就分配,那如何知道是大对象(存储在老生代)还是新对象(存储在新生代)?
作者回复: 声明变量是在编译阶段完成的,这时赋值语句还没执行! 比如 var a = 6 首先编译阶段确定有变量a了,并给a赋值undefined; 接下来执行代码,在执行过程中,会将6赋给a,这时候a等于6! 由于6是原生类型,通常情况下,会在栈上分配该变量! 如果 var a = Object 将对象赋给a时,在编译阶段 a依然等于undefined,在执行过程中,会在堆中创建一块内存,存放Object的值,然后栈中有个指向堆中Object地址的指针
2020-01-1315 - tick标记的过程具体是什么样的呢?我理解老师讲的是,一个指针指向堆里,每次移动一块内存,一个指针遍历栈中,然后看栈中是否引用这块堆中的内存,但感觉这样效率很低
作者回复: 比如全局window对象看成是一个树状结构,垃圾回收时,V8会先遍历这颗树,能遍历到的元素说明还存活的,标记为活动对象!没有被标记到的说明已经没有被引用了。 同时V8还维护了一个空闲列表,也就是没有被使用的空闲空间列表,垃圾清理过程就是把没有标记的添加到空闲列表中! 这样就完成了“标记-清除”操作
2019-09-03212 - 于你老师,我最近听了一门课,那个老师说现代的浏览器用闭包不会造成内存泄漏,因为垃圾回收是用的标记清除
作者回复: 对,没有被引用的闭包会被自动回收,不过如果没用的闭包还保存在全局变量中,依然会内存泄漏!
2019-09-07511