当前位置: 首页 > news >正文

前端知识点---垃圾回收机制(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 之间没有外部引用,它们会被视为不可达的对象,从而被清除。

相关文章:

  • mybatis-plus sql改写插件(二)
  • Android:Android Studio右侧Gradle没有assembleRelease等选项
  • 使用js脚本自动生成android项目的app icon图标
  • UML综合实验四
  • 【Linux】进程控制:创建、终止、等待与替换全解析
  • TurtleBot3 Package turtlebot3_drive source code read
  • web前端: 什么是web?
  • Linux shell脚本编程
  • <数据集>苹果识别数据集<目标检测>
  • Python标准库:sys模块深入解析
  • 增长黑客:技术与业务融合,驱动业务增长
  • 电商核心指标解析与行业趋势:数据驱动的增长策略【大模型总结】
  • ZeroLogon(CVE-2020-1472)漏洞复现
  • [蓝桥杯 2022 省 B] 李白打酒加强版
  • 工业制造各个系统术语
  • 提升Windows安全的一些措施
  • 死锁 手撕死锁检测工具
  • DDR中的delay line
  • doris基础使用
  • 汽车CAN总线采样点和采样率详解
  • 网站域名查询/新产品上市推广策划方案
  • 桂林 门户网站/百度竞价排名模式
  • 北京网站建设服务器维护/推广吧
  • 怎么把做的网站发到网上去/软文代写网
  • wordpress 位置/seo网站分析报告
  • 用自己电脑配置服务器做网站/网络营销的网站建设