欢迎来到尧图网

客户服务 关于我们

您的位置:首页 > 健康 > 美食 > 前端知识点---垃圾回收机制(javascript)

前端知识点---垃圾回收机制(javascript)

2025/4/21 7:18:17 来源:https://blog.csdn.net/2302_80171619/article/details/147077837  浏览:    关键词:前端知识点---垃圾回收机制(javascript)

JavaScript 垃圾回收机制

一、内存的生命周期

在 JavaScript 环境中,内存的一般生命周期如下:

  1. 内存分配:当我们声明变量、函数、对象时,系统会自动为它们分配内存。
  2. 内存使用:即对内存进行读写,也就是使用变量、函数等。
  3. 内存回收:使用完毕后,由垃圾回收器自动回收不再使用的内存。

💡 说明

  • 全局变量一般不会被回收,只有在关闭页面或刷新时才被释放。
  • 局部变量在作用域结束后,通常会被自动回收。

二、内存泄漏

内存泄漏是指:程序中分配的内存由于某些原因未被释放或无法释放,导致内存一直被占用。


三、堆和栈的区别(内存分配空间)

区域由谁分配存储内容特点
栈(Stack)操作系统自动分配函数参数、局部变量等基本数据类型生命周期短,效率高
堆(Heap)程序员手动分配(JS中由引擎处理)对象、数组等复杂数据类型生命周期长,需垃圾回收机制

在这里插入图片描述

四、垃圾回收机制 - 引用计数算法

IE 浏览器曾采用 引用计数(Reference Counting) 的垃圾回收算法:

🌟 基本原理:

  • 每个对象有一个引用计数值:
    1. 每当有一个引用指向该对象,计数器 +1。
    2. 每当一个引用被取消,计数器 -1。
    3. 当计数器为 0 时,表示该对象不再被使用,可以被回收。

✅ 示例:

let obj = { name: 'JS' }; // 创建了一个对象 { name: 'JS' } 变量 obj 指向这个对象  此时引用次数 = 1
let ref = obj;            // 引用数为 2
obj = 1;               // obj 不再引用那个对象了 引用次数 = 1(只有 ref 还在指向它)
ref = null;               //ref 也不再指向那个对象 此时引用次数 = 0

但它却存在一个致命的问题:嵌套引用(循环引用)

五 引用计数算法的设计缺陷

如果两个对象相互引用,尽管他们已不再使用,垃圾回收器不会进行回收,导致内存泄露。

因为他们的引用次数永远不会是0。这样的相互引用如果说很大量的存在就会导致大量的内存泄露

内存泄漏是指程序中已不再使用的内存未被释放,导致内存持续占用的现象。

你租了一个储物柜(内存),
放进去的东西(变量)用完了,
但你忘了把柜子钥匙还回去(释放内存),
所以柜子一直占着,别人也用不了,
时间久了,整个仓库都塞满了废东西

例子:

function fn()
let o1={}
let o2={}
o1.a=o2//.a只是 对象的属性,它并不需要在对象声明时预先定义o2.a=o1
return '引用计数没法回收'

在这里插入图片描述

六 标记清除法

现代的浏览器已经不再使用引用计数算法了,
现代浏览器通用的大多是基于标记清除算法的某些改进算法,总体思想都是一致的。
解决了技术算法的缺陷

核心:
1 标记清除算法将“不再使用的对象”定义为“无法达到的对象”
2 就是从根部(在JS中就是全局对象)出发定时扫描内存中的对象。凡是能从根部到达的对象,都是还需要使用的。
3 那些无法由根部出发触及到的对象被标记为不再使用,稍后进行回收。

在这里插入图片描述

🧠 标记清除法的具体步骤

1. 从根开始遍历

根对象通常是全局对象(如 windowglobal),以及当前活跃的变量、函数等。
对这些根对象进行遍历,标记所有可达的对象。

2. 标记所有可达对象

从根对象出发,访问每个可以直接或间接访问到的对象(即通过引用访问的对象)。
被访问到的对象会被标记为“存活”状态。

3. 清除不可达对象

完成标记后,所有没有被标记的对象都可以被认为是垃圾(不可达对象)。
这部分对象会被清除(回收内存)。

let obj1 = { name: 'Alice' };
let obj2 = { name: 'Bob' };
let obj3 = { name: 'Charlie' };// obj1 引用了 obj2
obj1.ref = obj2;// obj2 引用了 obj3
obj2.ref = obj3;// 根对象是 obj1

如果:

obj1 = null;

此时 obj1 被设为 null,它不再引用任何对象。由于 obj1 不再引用任何对象,而 obj2 和 obj3 之间没有外部引用,它们会被视为不可达的对象,从而被清除。

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com

热搜词