深入了解JavaScript中的垃圾回收机制

 更新时间:2023年04月11日 16:41:41   作者:施主来了  
JavaScript中的垃圾回收机制负责自动管理内存,回收不再使用的对象所占用的内存空间。本文主要介绍了JS中垃圾回收机制的相关知识,需要的可以参考一下

JavaScript中的垃圾回收机制负责自动管理内存,回收不再使用的对象所占用的内存空间。在JavaScript中,开发者不需要显式地分配和释放内存,垃圾回收器会自动完成这些操作。以下是关于JavaScript垃圾回收机制的一些关键概念:

  • 内存生命周期:JavaScript内存生命周期包括分配、使用和释放三个阶段。首先,内存会被分配给变量或对象;然后,程序会使用这些变量或对象;最后,不再需要的变量或对象会被垃圾回收器释放。
  • 可达性:垃圾回收器通过可达性来判断一个对象是否还在使用。根对象(如全局对象和其他内置对象)被认为是可达的。如果一个对象可以通过根对象或其他可达对象引用链到达,那么它也被认为是可达的。
  • 引用计数:这是一种较早的垃圾回收策略,通过追踪每个对象的引用次数来判断对象是否仍在使用。当对象的引用计数为0时,表示对象不再被使用,可以被回收。然而,引用计数算法存在循环引用问题,无法回收循环引用的对象。
  • 标记-清除:这是现代JavaScript引擎中常见的垃圾回收算法。标记-清除算法首先会标记所有可达对象,然后遍历整个内存空间,清除未被标记的对象。这种算法可以处理循环引用问题,但可能会导致内存碎片。
  • 分代回收:由于不同对象的生命周期长短不同,现代JavaScript引擎将内存分为新生代和老生代。新生代主要存放短生命周期的对象,老生代主要存放长生命周期的对象。新生代和老生代的垃圾回收策略会有所不同。
  • 增量回收和懒惰回收:为了降低垃圾回收对程序执行的影响,现代JavaScript引擎采用了增量回收和懒惰回收策略。增量回收将回收工作分成多个小任务,穿插在程序执行过程中;懒惰回收则会在一定程度上推迟回收操作,以减少性能开销。

以下是一个简单的示例,演示了 JavaScript 垃圾回收机制中的引用计数和标记清除:

// 引用计数示例
let a = { name: 'John' };
let b = a; // b 引用了 a,a 的引用计数变为 2
a = null; // a 不再引用这个对象,a 的引用计数变为 1
b = null; // b 不再引用这个对象,这个对象的引用计数变为 0,可以被垃圾回收器回收

// 标记清除示例
function foo() {
  let x = { name: 'Alice' };
  let y = { name: 'Bob' };
  x.friend = y;
  y.friend = x;
}

foo(); // 函数执行完后,x 和 y 都不再被使用,但它们之间相互引用,无法使用引用计数来回收内存
// 垃圾回收器定期运行,会发现 x 和 y 都已经不再被引用,可以被回收

在这个示例中,当变量 a 被赋值给变量 b 时,对象的引用计数变为 2。当 a 被赋值为 null 时,对象的引用计数变为 1。最后当 b 也被赋值为 null 时,对象的引用计数变为 0,可以被垃圾回收器回收。

另外,函数 foo 中创建了两个对象 xy,并且它们相互引用。在函数执行完后,这两个对象不再被使用,但它们之间的引用关系无法使用引用计数来回收内存。因此,垃圾回收器会定期运行,查找那些已经不再被引用的对象,然后释放它们所占用的内存空间。

再来一个例子,我们将创建一些对象并解释JavaScript的垃圾回收机制。

// 创建对象
function createPerson(name, age) {
    return {
        name: name,
        age: age,
    };
}

// 创建两个对象
let person1 = createPerson("Alice", 30);
let person2 = createPerson("Bob", 35);

// person1 和 person2 变量引用了两个新创建的对象,这些对象在内存中是可达的

// 现在将 person1 引用另一个对象
person1 = createPerson("Charlie", 28);

// 之前 person1 引用的 "Alice" 对象现在已经不再可达,因为没有变量引用它
// JavaScript的垃圾回收器会识别到这一点,并在合适的时机释放其内存

// 创建一个循环引用
let objA = {
    name: "ObjA",
};
let objB = {
    name: "ObjB",
};
objA.link = objB;
objB.link = objA;

// 将变量设置为 null,打破可达性
objA = null;
objB = null;

// 现在 objA 和 objB 对象都不再可达,即使它们彼此引用
// 使用标记-清除算法的垃圾回收器会识别到这一点,并释放它们占用的内存

在这个例子中,我们创建了几个对象并对它们进行了引用。当一个对象不再可达时,它就成为了垃圾回收的目标。对于循环引用的情况,标记-清除算法可以识别到并正确处理这种情况,释放不再使用的对象所占用的内存。

注意:不同的JavaScript引擎可能采用不同的垃圾回收策略,如V8、SpiderMonkey和JavaScriptCore等。

到此这篇关于深入了解JavaScript中的垃圾回收机制的文章就介绍到这了,更多相关JavaScript垃圾回收机制内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • js调用网络摄像头的方法

    js调用网络摄像头的方法

    这篇文章主要介绍了js调用网络摄像头的方法,帮助大家更好的理解和使用JavaScript,感兴趣的朋友可以了解下
    2020-12-12
  • 关于Mozilla浏览器不支持innerText的解决办法

    关于Mozilla浏览器不支持innerText的解决办法

    在各大浏览器中,除Mozilla浏览器外,几乎都支持一个元素的属性:innerText。我们可以通过它来快速获取某个元素的内的文本。
    2011-01-01
  • JS处理一些简单计算题

    JS处理一些简单计算题

    本篇文章通过JS来实现以下日常用经常用到的计算以及换算等小工具,有兴趣的朋友可以学习下。
    2018-02-02
  • js中关于Blob对象的介绍与使用

    js中关于Blob对象的介绍与使用

    这篇文章主要介绍了js中关于Blob对象的介绍与使用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-11-11
  • 由 element.appendChild(newNode) ,谈开去

    由 element.appendChild(newNode) ,谈开去

    由 element.appendChild(newNode) ,谈开去...
    2006-11-11
  • js读写json文件实例代码

    js读写json文件实例代码

    json文件想必大家对它并不陌生吧,它的灵活性还是不错的,下面为大家介绍下js如何读写json文件
    2014-10-10
  • WebSocket中心跳检测与断开重连机制详解

    WebSocket中心跳检测与断开重连机制详解

    心跳检测是指在WebSocket连接过程中定时向服务端发送和接收心跳消息,来确定当前连接是否是正常状态的检测机制,断开重连是指在WebSocket不正常断开连接后,进行重新连接的策略,下面我们就来看看这二者的具体实现吧
    2024-01-01
  • 微信小程序实现多选框全选操作

    微信小程序实现多选框全选操作

    这篇文章主要为大家详细介绍了微信小程序实现多选框全选操作,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-07-07
  • jsonp跨域获取数据的基础教程

    jsonp跨域获取数据的基础教程

    这篇文章主要给大家介绍了关于jsonp跨域获取数据的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2018-07-07
  • 使用OPENLAYERS3实现点选的方法

    使用OPENLAYERS3实现点选的方法

    这篇文章主要为大家详细介绍了使用OPENLAYERS3实现点选的几种方法,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-07-07

最新评论