闭包、内存泄漏、垃圾回收详解

news2025/1/17 2:19:49

首先要说清楚这个话题,必须要先清楚什么是垃圾回收,要清楚什么是垃圾回收呢,必须要知道什么是垃圾,所谓的垃圾就是不再需要的内存,需要或者不需要是由人为来决定的

<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <meta name="viewport" content="width=device-width, initial-scale=1.0" />
    <title>Document</title>
  </head>
  <body>
    <button>点击</button>
  </body>
  <script>
    const createIncrease = () => {
      const doms = new Array(10000).fill(0).map((_, i) => {
        const dom = document.createElement("div");
        dom.innerHTML = 1;
        return dom;
      });
      const increase = () => {
        doms.forEach((e) => {
          e.innerHTML = Number(e.innerHTML) + 1;
        });
      };
      return increase;
    };
    const increase = createIncrease();
    const btn = document.querySelector("button");
    btn.addEventListener("click",increase);
  </script>
</html>

就比如上面的代码,这个doms很明显是需要的内存,因为每次点击按钮都会执行increase这个函数,这个函数里面用到了doms,所以doms不是垃圾

 const nums = [1,2,3,4,5];
 const sum = nums.reduce((pre,next) =>{
     return pre + next
 },0);
console.log(sum);

上面的nums是不是垃圾呢,不是垃圾,虽然看起来这三行代码运行结束后,没有再需要nums的地方了,按道理是不再需要的东西了,但是可以在浏览器的控制台打印这个nums,所以它不是垃圾
在这里插入图片描述
所以需不需要得问自己,毕竟是我们在写代码,只有我们清楚后续代码还需不需要,所以垃圾就是我们能清楚知道后续不再使用的内存
现在搞清楚了垃圾,那么再来搞清楚垃圾回收,js里面是有一个垃圾回收器来帮助我们回收不再需要的内存,但是这个玩意儿它根本不知道这个内存需不需要,就比如上面的nums不也没有回收嘛,但是它知道有一些东西一定是我们不需要的,那就是连我们自己都访问不到的内存,举个例子

  let nums = [1,2,3];
  nums = [4,5];
  const sum = nums.reduce((pre,next) =>{
      return pre + next
  },0);
  console.log(sum);

nums被重新赋值后,很明显,数组[1,2,3]的内存我们已经没有任何机会再访问到了,被称为无法触达的内存空间,这一类内存就会被垃圾回收器定义为垃圾,这就是垃圾回收
那什么是内存泄漏呢,就是我们不再需要使用的内存空间依旧能够触达,导致垃圾回收器并不能将其回收,也就是上面例子中的nums,当内存泄漏过多的时候,就会影响代码的运行,因此需要手动将其变成无法触达的内存空间,操作很简单,在代码最后将nums赋值为null,那么数组[1,2,3]的内存空间将变成无法触达,也就会被垃圾回收器回收了

 let nums = [1,2,3];
 const sum = nums.reduce((pre,next) =>{
    return pre + next
 },0);
 nums = null;
 console.log(sum);

所以不仅仅是闭包才会造成内存泄漏,正常定义了变量最后没有设置为null也会导致内存泄漏
闭包内存泄漏的原因主要有两点:

  • 持有了不再需要的函数引用,会导致函数关联的词法环境无法销毁,从而导致内存泄漏
  • 当多个函数共享词法环境时,会导致词法环境膨胀,从而导致出现无法触达但也无法回收的内存空间,从而导致内存泄漏
<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <meta name="viewport" content="width=device-width, initial-scale=1.0" />
    <title>Document</title>
  </head>
  <body>
    <button>点击</button>
  </body>
  <script>
    const createIncrease = () => {
      const doms = new Array(10000).fill(0).map((_, i) => {
        const dom = document.createElement("div");
        dom.innerHTML = 1;
        return dom;
      });
      const increase = () => {
      };
      const test = () =>{
        doms
      }
      return increase;
    };
    const increase = createIncrease();
    const btn = document.querySelector("button");
    btn.addEventListener("click",() =>{
        increase()
    });
  </script>
</html>

上面的例子中,increase函数并没有使用doms,test函数中使用了,但是test函数根本访问不到,但是因为它和increase函数共享词法环境,导致doms即使是无法触达的内存空间依旧无法被回收

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1830820.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

原生APP和H5 APP的区别

原生APP和H5 APP是两种常见的移动应用开发方式。它们在技术架构、性能表现、开发成本、用户体验等方面都有着明显的区别。北京木奇移动技术有限公司&#xff0c;专业的软件外包开发公司&#xff0c;欢迎交流合作。 1. 技术架构 原生APP&#xff1a;原生APP是使用手机操作系统的…

表格识别工具哪个好?简单操作,一键识别表格

随着2024年高考的圆满结束&#xff0c;考生们迎来了新的挑战——志愿填报。这不仅是一个技术活&#xff0c;更是一个信息战。 面对海量的高校信息和复杂的数据表格&#xff0c;考生们需要一种快速、准确的方法来整理和分析这些数据。幸运的是&#xff0c;现代科技提供了多种表…

CPRI协议理解——控制字内容

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 CPRI协议理解——控制字内容 前言同步标识L1 Inband ProtocolZ130.0Z.194 C&M 通道慢速C&M 通道快速C&M 通道Vendor Specific DataControl AxC Data 后记 前言 …

m4s转mp3——B站缓存视频提取音频

前言 しかのこのこのここしたんたん&#xff08;鹿乃子乃子虎视眈眈&#xff09;非常之好&#xff0c;很适合当闹钟&#xff0c;于是缓存了视频&#xff0c;想提取音频为mp3 直接改后缀可乎&#xff1f;格式转换工具&#xff1f; 好久之前有记录过转MP4的&#xff1a; m4s转为…

DbGate 开源、免费的 、智能的、NoSQL SQL 数据库工具

1、简介 DbGate 是麻省理工学院许可的开源项目。支持Windows、Linux、MacOS、WEB(Docker、NPM) 2、下载地址/官网 DbGate | Open Source SQLnoSQL Database Client Tips&#xff1a;下载时建议使用稳定版本哦&#xff01; 3、支持的数据库 MySQL、SQLServer、Oracle、Postg…

【C++提高编程-08】----C++ STL之常用查找算法

&#x1f3a9; 欢迎来到技术探索的奇幻世界&#x1f468;‍&#x1f4bb; &#x1f4dc; 个人主页&#xff1a;一伦明悦-CSDN博客 ✍&#x1f3fb; 作者简介&#xff1a; C软件开发、Python机器学习爱好者 &#x1f5e3;️ 互动与支持&#xff1a;&#x1f4ac;评论 &…

实用技巧,用lsof命令监控tar文件解压进度,简单有效!

在Linux系统中&#xff0c;tar命令是一个非常常用的工具&#xff0c;用于创建和解压缩归档文件。尽管tar命令本身没有提供直接查看解压进度的功能&#xff0c;但我们可以借助lsof&#xff08;List Open Files&#xff09;命令来间接监控解压进度。本文将详细介绍如何使用lsof命…

C#:ThreadPool 实现高效多线程处理

请关注微信公众号&#xff1a;拾荒的小海螺 博客地址&#xff1a;http://lsk-ww.cn/ 1、简述 在现代软件开发中&#xff0c;提升应用程序的并发能力和性能是一个重要的任务。C# 提供了多种实现并发的方式&#xff0c;其中&#xff0c;ThreadPool&#xff08;线程池&#xff0…

亚马逊新店如何实现高效流量转化?自养号测评深度解析与实用策略

在亚马逊平台上&#xff0c;自养号测评是一种通过卖家自行控制的海外买家账号对商品进行评价的方法&#xff0c;旨在提高商品的排名和流量。 亚马逊的自养号测评是指卖家通过使用在海外真实环境注册的买家账号&#xff0c;代替真实买家对商品进行测评。账号由卖家自己管理&…

Python对象复制竟然有这么多种方式,赶紧学起来!

目录 1、浅拷贝:copy模块的copy()函数 📋 1.1 浅拷贝原理揭秘 1.2 实战演示:列表与字典的浅拷贝 列表浅拷贝示例 字典浅拷贝示例 1.3 注意事项:共享引用与独立对象 2、深拷贝:copy模块的deepcopy()函数 📌 2.1 深拷贝实现机制解析 2.2 深拷贝优势分析 2.3 深度…

SRM供应商管理系统建设方案及源码实现(方案+源码)

1. 供应商管理 2. 采购需求管理 3. 采购寻源管理 4. 采购合同管理 5. 采购订单管理 6. 采购协同管理 7. 外部商城采购管理 8. 报表查询管理 9. 系统管理 10. 集成管理 资料获取&#xff1a;本文末个人名片。

了解压电传感器:压电效应

压电加速度计的个关键方面是压电效应。一般来说&#xff0c;压电材料在受到机械应力时可以产生电力。 相反&#xff0c;对压电材料施加电场可以使其变形并产生小的机械力。尽管大多数电子工程师都熟悉压电效应&#xff0c;但有时并没有完全理解这种有趣现象的细节。 更深入地…

visualbox搭建linux环境双网卡配置

文章目录 1. 双网卡模式简介2. 网络模式配置2.1 virtualBox说明2.2 host-only网络模式配置2.3 NAT网络模式配置 3. 虚拟主机网络设置3.1 网卡一设置3.2 网卡二设置 4. 网卡配置5. ssh访问 本篇的目的是为了搭建本地的linux测试环境用。 1. 双网卡模式简介 双网卡网络模式简介 …

期货交易如何定义趋势?

从任何交易周期来看&#xff0c;行情无非就处在趋势、震荡这两种情况中。如果我们再把一个新的趋势突破原来的状态&#xff0c;逐步成为有力量的趋势&#xff0c;叫做反转&#xff0c;那么可以有三个形态&#xff0c;即趋势形态、震荡形态、反转形态。 做交易的人&#xff0c;…

消息队列-RabbitMQ-延时队列实现

死信队列 DLX,全称为Dead-Letter-Exchange,死信交换机&#xff0c;死信邮箱。当消息在一个队列中变成死信之后&#xff0c;它能重新发送到另外一个交换器中&#xff0c;这个交换器就是DLX&#xff0c;绑定DLX的队列就称为死信队列。 导致死信的几种原因&#xff1a; ● 消息…

GIS开发程序员简历怎么写?

GIS开发或者是数据处理等相关岗位&#xff0c;其本质还是开发类技术岗位。所以怎么让你的简历脱颖而出&#xff0c;最最重要的就是&#xff0c;简历上要展现两点&#xff1a;你有能力能胜任此岗位和你有潜力可以快速胜任此岗位&#xff08;校招上也看中这点&#xff09;。 那G…

轨迹优化 | 图解欧氏距离场与梯度场算法(附ROS C++/Python实现)

目录 0 专栏介绍1 什么是距离场&#xff1f;2 欧氏距离场计算原理3 双线性插值与欧式梯度场4 仿真实现4.1 ROS C实现4.2 Python实现 0 专栏介绍 &#x1f525;课程设计、毕业设计、创新竞赛、学术研究必备&#xff01;本专栏涉及更高阶的运动规划算法实战&#xff1a;曲线生成…

Easyui Datagrid 解决页面加载重复请求2次后端问题

现象 在页面刚刚打开时&#xff0c;页面加载完毕&#xff0c;显示查询数据&#xff0c;连续请求网络2次&#xff0c; 第一次是只携带了 分页参数&#xff1b; 第二次携带分页和查询参数 问题出现原因 html代码中利用class声明了datagrid&#xff0c;导致easyUI解析class代…

【Android】安卓开发的前景

人不走空 &#x1f308;个人主页&#xff1a;人不走空 &#x1f496;系列专栏&#xff1a;算法专题 ⏰诗词歌赋&#xff1a;斯是陋室&#xff0c;惟吾德馨 目录 &#x1f308;个人主页&#xff1a;人不走空 &#x1f496;系列专栏&#xff1a;算法专题 ⏰诗词歌…

PyQt5 生成py文件不能运行;pushButton点击事件;QTextEdit 获取输入框内容

目录 cant open file c.pyuic: c.pyuic $FileName$ -o $FileNameWithoutExtension$.p PyQt5 生成py文件不能运行 pushButton点击事件 QTextEdit 获取输入框内容 整体运行代码: Creating a Qt Widget Based Application | Qt Creator Manual cant open file c.pyuic: c.…