浅谈内存泄漏

news2025/1/23 3:54:01

内存泄漏

概念

在JavaScript中,内存泄漏是指应用程序在不再需要使用某块内存时仍然保持对其的引用,导致内存不能被垃圾回收机制释放,最终导致内存占用过高,性能下降。

内存泄漏通常发生在以下情况:

  • 全局变量:全局变量会一直存在于内存中,即使在不需要时也无法被垃圾回收机制回收。
    未清理的定时器和回调函数:如果定时器或回调函数没有正确清理,它们将一直存在于内存中,即使它们已经完成或不再需要了。
  • DOM引用:当从DOM中删除元素时,如果仍然保持对该元素的引用,它将会在内存中留下无法释放的引用。
  • 闭包:如果函数创建了一个闭包(即函数内部引用了外部函数的变量),并且闭包没有在正确的时机释放,这将导致外部函数的变量一直无法被垃圾回收。

为了避免内存泄漏,可以采取以下措施:

  • 使用局部变量:在函数内部尽量使用局部变量而不是全局变量,以便在函数执行完毕后可以被垃圾回收。
  • 显式清理:在不再需要使用的定时器、回调函数和DOM引用时,手动清理它们,例如通过使用clearTimeout()清除定时器。
  • 解除闭包:在不需要使用闭包时,手动解除对外部函数变量的引用,以便垃圾回收机制可以释放内存。
  • 使用垃圾回收器:JavaScript具有自动的垃圾回收机制,可以自动释放不再需要的内存。通过确保代码的正确性和效率,可以帮助垃圾回收器更好地工作。

常见的内存泄漏

全局变量

  1. 意外的全局变量

    function leakFunction() {
      leakVar = 'test'
    }
    

    外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

    解决方法use strict 启用严格模式,严格模式会阻止创建意外的全局变量

  2. 全局变量未释放

    全局变量未释放也会导致内存泄漏,可通过将变量置为null释放全局变量,避免内存泄漏

    // 未释放的全局变量
    var globalVar = 'Hello'
    
    function doSomething() {
    // ...
    }
    
    // 忘记释放全局变量
    
    // 可通过以下方式释放
    // globalVar = null
    // doSomething = null
    

闭包

闭包 中被访问的参数和变量不会被JavaScript的垃圾回收机制回收,会导致内存泄漏,可通过将变量置为null释放全局变量,避免内存泄漏

function createClosure() {
  var value = 'Hello'

  // 创建一个闭包函数
  function closureFunction() {
    console.log(value)
  }

  // 返回闭包函数
  return closureFunction
}

// 创建闭包
var closure = createClosure()

// 闭包持有外部变量的引用,可能导致内存泄漏

// 可通过以下方式释放
// closure = null

定时器未清理

未清理定时器也会导致内存泄漏,可通过clearInterval清理定时器,避免内存泄漏

// 创建一个定时器
var timer = setInterval(function() {
  // 定时器回调函数
  // ...
}, 1000)

// 定时器未清理

// 可通过以下方式清理定时器
// clearInterval(timer)

清除页面dom元素不完全

  1. 清除页面dom元素时,dom元素绑定的事件未解绑
var element = document.getElementById('button')
var counter = 0
function onClick(event) {
  counter++
  element.innerHtml = 'clickTime ' + counter
}

element.addEventListener('click', onClick)


// 可通过以下方式清理
// element.removeEventListener('click', onClick)
// element.parentNode.removeChild(element)

循环引用

当两个或多个对象之间存在相互引用,并且没有被其他对象引用,就会发生循环引用,从而导致内存泄漏。这种情况可以通过在对象之间断开引用来避免

function createObject() {
  var obj1 = {}
  var obj2 = {}
  obj1.ref = obj2
  obj2.ref = obj1
  return obj1
}
var myObj = createObject()
// 这里无法回收 myObj 和 myObj.ref 所占用的内存空间,导致内存泄漏

常见的防止内存泄露方法

  1. 显式释放引用

    在不再使用变量、对象或函数时,手动将其引用置为null,以便垃圾回收器可以正确地回收内存

    var obj = {}
    
    // 使用obj
    
    // 当不再需要obj时,将其引用置为null
    obj = null
    
  2. 清除定时器

    在使用setInterval或setTimeout函数创建定时器时,应确保在不需要时及时清除定时器。可以使用clearInterval或clearTimeout函数来清除定时器

    var timer = setInterval(function() {
    // 定时器回调函数
    // ...
    
    // 当不再需要定时器时,清除定时器
    clearInterval(timer)
    }, 1000)
    
  3. 移除事件监听器

    在使用addEventListener函数添加事件监听器时,确保在不需要时移除事件监听器,以避免对象无法被垃圾回收。可以使用removeEventListener函数来移除事件监听器

    var button = document.createElement('button')
    
    function handleClick() {
    // 点击事件回调函数
    // ...
    
    // 当不再需要事件监听器时,移除事件监听器
    button.removeEventListener('click', handleClick)
    }
    
    // 添加点击事件监听器
    button.addEventListener('click', handleClick)
    
  4. 避免创建不必要的闭包

    闭包可以使函数保持对其定义时的作用域的引用,从而可能导致内存泄漏。尽量避免在不需要时创建不必要的闭包

    function createClosure() {
      var value = 'Hello'
      // 创建一个闭包函数
      function closureFunction() {
        console.log(value)
      }
      // 返回闭包函数
      return closureFunction
    }
    
    // 创建闭包
    var closure = createClosure()
    
    // 当不再需要闭包时,释放闭包引用
    closure = null
    

相关文档

【温故而知新】JavaScript中内存泄露有那几种

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

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

相关文章

函数递归练习

目录 1.分析下面选择题 2.实现求第n个斐波那契数 3.编写一个函数实现n的k次方,使用递归实现。 4.写一个递归函数DigitSum(n),输入一个非负整数,返回组成它的数字之和 5.递归方式实现打印一个整数的每一位 6.实现求n的阶乘 1.分析下面选择…

linux Docker在线/离线服务安装并支持centos7和centos8系统

注:以下内容都是经过测试;能在生产环境使用. 一、centos7版本的docker在线安装 1:运行以下命令,下载docker-ce的yum源。 sudo wget -O /etc/yum.repos.d/docker-ce.repo https://mirrors.aliyun.com/docker-ce/linux/centos/docker-ce.repo…

外卖系统拦截器实现(Interceptor)

SpringMVC的拦截器主要是用于拦截控制器方法的执行; 概念:是一种动态拦截方法调用的机制,类似于过滤器。在Spring中动态拦截控制器中方法的执行。 作用:在指定的控制器中调用前后执行预先设定的代码,完成功能增强。 应…

人才测评:计划管理能力与岗位胜任力素质测评

计划管理能力指的是什么? 计划管理能力,可以体现为从业者在精准制定好任务,或是根据任务的时间长,困难的程度来设定好完成的目标,一步一个脚印将工作完成好,并且能预估出可能出现的突发事件,将…

STM32--HC-SR501 热释电人体红外感应模块

实物引脚图: 模块工作特性: 当人进入感应范围之后输出引脚输出高电平,人离开感应范围自动延时输出低电平 热释电效应: 热释电传感器,也称为人体红外传感器,其工作原理基于热释电效应。这种传感器由几个关…

2024护网在即,今年的护网招聘着实有点离谱了!

最近刷到条护网招聘的朋友圈,今年的护网待遇着实有点离谱了,日薪提到了1w,还是全款垫付???!! 我算是理解了“我们当年可没你现在这个条件”这句话。 先给大家科普下护网行动&#x…

【C语言】6.C语言VS实用调试技巧(2)

文章目录 6.调试举例17.调试举例28.调试举例3&#xff1a;数组9.编程常⻅错误归类9.1 编译型错误9.2 链接型错误9.3 运⾏时错误 – 6.调试举例1 求 1!2!3!4!…10! 的和。 int main() {int n 0;int i 0;int ret 1;int sum 0;for (n 1; n < 3; n) {for (i 1; i < …

数据结构之栈与队列

一.栈 1.定义&#xff1a;一种线性表&#xff0c;只允许在固定的一端进行删除和插入数据的操作&#xff0c;该端叫栈底&#xff0c;另一端叫栈顶 2.特点&#xff1a;先进后出 注&#xff1a;栈中元素出栈是一对多的(他虽然满足先进后出但是我们可以在pop数据前先获取栈顶元素…

Tower for Mac:Git管理的新境界

Tower for Mac&#xff0c;让您的Git管理进入新境界&#xff01;这款专为Mac用户打造的Git客户端&#xff0c;凭借其出色的性能和丰富的功能&#xff0c;成为众多开发者的首选工具。 Tower不仅支持常规的Git操作&#xff0c;如提交、推送和拉取&#xff0c;还提供了许多高级功能…

【微服务最全详解】

文章目录 微服务微服务的介绍微服务服务架构演变 微服务网关微服务的负载均衡微服务的容灾机制服务崩溃服务容灾机制微服务熔断机制微服务限流Sentinel怎么实现限流微服务限流算法1.令牌桶算法2.漏斗桶算法 服务监控日志收集 微服务 微服务的介绍 微服务是一种软件架构风格&a…

MyBatis的创建和测试

创建项目点击Spring Initializr然后点击next 点击SQL 选择里面的Mybatis Framework和Mysql Driver 按如下图片创建项目 user表中的数据 #下面这些内容是为了让MyBatis映射 #指定Mybatis的Mapper文件 mybatis.mapper-locationsclasspath:mappers/*xml #指定Mybatis的实体目录 my…

使用人人开源renren-fast快捷搭建后台管理系统

https://gitee.com/renrenio/renren-fast https://gitee.com/renrenio/renren-fast 初始化项目数据库 导入项目运行 期间遇到的坑 024-04-25 01:30:27.638 ERROR 25228 --- [ main] com.alibaba.druid.pool.DruidDataSource : init datasource error, url: jdbc:…

【安全每日一讲】API是什么?解密API背后的奥秘

什么是API? API全称Application Programming Interface&#xff0c;即应用程序编程接口&#xff0c;是一些预先定义的函数&#xff0c;或指软件系统不同组成部分衔接的约定&#xff0c;用于传输数据和指令&#xff0c;使应用程序之间可以集成和共享数据资源。 简单来说&#…

thinkphp8 framework和 element plus admin前后端分离系统之PHP安装教程

DIYGW-UI-PHP是一款基于thinkphp8 framework和 element plus admin开发而成的前后端分离系统。目的是结合现有diygw-ui打造一个后台API开发。 实现PHP源码前请先下载小皮面板或者宝塔。 系统已经集成了部分功能 用户管理 后台用户管理部门管理 配置公司的部门结构&#xff0…

用红黑树封装出map与set

目录 一、红黑树的改造 节点结构的定义 迭代器类的实现 红黑树中提供迭代器 红黑树的主要代码 二、set的实现 三、map的实现 四、测试代码 map与set的底层都是红黑树&#xff0c;所以本篇文章就分享如何用同一颗红黑树封装出map与set 所以大家可以先去看一下我的讲解红…

先有JVM还是先有垃圾回收器?很多人弄混淆了

是先有垃圾回收器再有JVM呢&#xff0c;还是先有JVM再有垃圾回收器呢&#xff1f;或者是先有垃圾回收再有JVM呢&#xff1f;历史上还真是垃圾回收更早面世&#xff0c;垃圾回收最早起源于1960年诞生的LISP语言&#xff0c;Java只是支持垃圾回收的其中一种。下面我们就来刨析刨析…

ue引擎游戏开发笔记(38)——实现敌人接收攻击伤害,并作出反应

1.需求分析&#xff1a; 现在已经显示造成实际伤害&#xff0c;但敌人对实际伤害并未产生反馈&#xff0c;例如还击&#xff0c;或者死亡倒地等等&#xff0c;实现敌人对于受击的反馈。 2.操作实现&#xff1a; 1.思路&#xff1a;在动画蓝图中添加死亡动画&#xff0c;并通过…

wefaf

c语言中的小小白-CSDN博客c语言中的小小白关注算法,c,c语言,贪心算法,链表,mysql,动态规划,后端,线性回归,数据结构,排序算法领域.https://blog.csdn.net/bhbcdxb123?spm1001.2014.3001.5343 给大家分享一句我很喜欢我话&#xff1a; 知不足而奋进&#xff0c;望远山而前行&am…

Python 渗透测试:反弹 shell (反弹 后门 || 程序免杀)

什么叫 反弹 shell 反弹 shell (Reverse Shell) 是一种常见的渗透测试技术,它指的是受害者主机主动连接攻击者的主机,从而让攻击者获得对受害者主机的控制权。在一个典型的反弹 shell 攻击中,攻击者会在自己的主机上监听一个特定的端口,然后诱使目标主机主动连接到这个端口。当…

Cadence 16.6 绘制PCB封装时总是卡死的解决方法

Cadence 16.6 绘制PCB封装时总是卡死的解决方法 在用Cadence 16.6 PCB Editor绘制PCB封装时候&#xff0c;绘制一步卡死一步&#xff0c;不知道怎么回事儿&#xff0c;在咨询公司IT后&#xff0c;发现是WIN系统自带输入法的某些热键与PCB Editor有冲突&#xff0c;导致卡死。 …