写点什么

前端内存泄漏详解

作者:不叫猫先生
  • 2023-06-07
    北京
  • 本文字数:1712 字

    阅读完需:约 6 分钟

前端内存泄漏详解

🥙一、什么是内存泄漏

JavaScript 通过自动内存管理实现内存分配和闲置资源回收。基本思路很简单:确定哪个变量不会再使用,然后释放它占有的内存。这个过程是周期性的,即垃圾回收程序每隔一定时间(或者说在代码执行过程中某个预订的收集时间)就会自动运行。垃圾回收是一个近似且不完美的方案,因为某块内存是否还有用,属于“不可判定的”问题,意味着靠算法是解决不了的。——《JavaScript 高级程序设计(第 4 版)》4.3 垃圾回收


我们知道了 JS 对内存管理是自动的,并没特殊的机制去实现。那么为什么有时候会出现内存泄漏的情况呢?主要原因在于应用程序分配内存之后,由于程序设计错误,导致无法对分配的内存进行管理,无法垃圾回收(GC)、释放内存,情况严重则会导致系统卡死。==<font color=red >内存泄漏就是未能释放不在使用的内存</font>==


垃圾回收过程是不实时进行的,因为 JavaScript 是一门单线程的语言,每次执行垃圾回收,会使程序应用逻辑暂停,执行完垃圾后回收再执行应用逻辑,这种行为称为全停顿,所以 ==<font color=red >一般垃圾回收会在 cpu 闲时进行</font>== 。

🥪二、内存生命周期

我们在创建变量、函数或者其他任何内容的时候,JS 引擎会自动为我们分配内存,并且在不需要的时候释放内存。一共需要经历三个阶段:


  • 内存分配:当我们创建变量或函数时,JS 引擎会为我们分配一些内存空间来存放该变量的内容

  • 内存使用:使用分配得到内存,就可以在 js 中读取并写入变量或者对象的属性值

  • 内存释放:在不需要变量或者函数时候,JS 引擎会自动清除(闭包、程序 bug 除外)


当然内存分配包括了静态分配和动态分配,我们在这里暂且不谈论。

🌮三、导致内存泄漏可能存在的情况以及解决方法

  • addEventListeneraddEventListener 添加在全局变量比如:window、body 等时,组件销毁时,就会导致内存泄漏;如果在组件 dom 上进行监听便不会导致内存泄漏,因为 dom 销毁时候监听器会自动移除。使用 addEventListener 导致内存泄漏时,需要使用 removeEventListener 移除。

  • setTimeout/setTimeInterval 使用定时器时候一定要记得在不需要使用时候用 clearTimeout、clearInterval 清除掉。

  • URL.createObjectURL 每次调用 URL.createObjectURL 时候都会创建一个新的对象,在不需要使用该对象的时候,一定要用 URL.revokeObjectURL()清除掉创建的对象。


  let url = window.URL.createObjectURL(new Blob([data]))  let link = document.createElement('a')  link.style.display = 'none'  link.href = url  link.setAttribute('download', filename)  document.body.appendChild(link)  link.click()  document.body.removeChild(link); //下载完成移除元素  window.URL.revokeObjectURL(url); //释放掉blob对象
复制代码


  • 不正当的闭包情况


function bibao(){    let a = 0;    return function(){        return a    }}let func = bibao()func()
复制代码


return 的函数中对 bibao 函数中的 a 变量有引用,故而 a 并不会被垃圾回收,造成内存泄漏,解决办法为当不使用 bibao 函数时,将 func 置空:


function bibao(){    let a = 0;    return function(){        return a    }}let func = bibao()func()func = null;
复制代码


另外,全局变量可以重复使用,但是容易造成变量污染。不同的地方定义了相同的全局变量,这样就会产生混乱。局部变量仅在局部作用域内有效,不可以重复使用,不会造成变量污染。而闭包结合了全局变量和局部变量的优点,==<font color=red >可以重复使用变量,并且不会造成变量污染</font>==


  • 隐式全局变量全局变量除非被取消或者重新分配之外也是无法回收的。


function ImplicitGlobalVariables(){    a= 0;    this.b = 3;}
复制代码


  • 子元素存在引用


div id="root">  <ul id="ul">    <li></li>    <li id="li"></li>  </ul></div><script>  let root = document.querySelector('#root')  let ul = document.querySelector('#ul')  let li = document.querySelector('#li')    // 由于ul变量存在,整个ul及其子元素都不能被回收  root.removeChild(ul)  // 虽置空了ul变量,但由于li3变量引用ul的子节点,所以ul元素依然不能被回收  ul = null  // 已无变量引用,此时可以回收  li = null</script>
复制代码


发布于: 刚刚阅读数: 4
用户头像

还未添加个人签名 2022-10-18 加入

前端领域优质创作者、阿里云专家博主,专注于前端各领域技术,共同学习共同进步,一起加油呀!

评论

发布
暂无评论
前端内存泄漏详解_内存泄露_不叫猫先生_InfoQ写作社区