iis服务器助手广告广告
返回顶部
首页 > 资讯 > 精选 >怎么避免JavaScript内存泄漏
  • 899
分享到

怎么避免JavaScript内存泄漏

2023-06-30 17:06:40 899人浏览 薄情痞子
摘要

这篇文章主要介绍“怎么避免javascript内存泄漏”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“怎么避免JavaScript内存泄漏”文章能帮助大家解决问题。一、什么是内存泄漏JavaScrip

这篇文章主要介绍“怎么避免javascript内存泄漏”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“怎么避免JavaScript内存泄漏”文章能帮助大家解决问题。

一、什么是内存泄漏

JavaScript 就是所谓的垃圾回收语言之一,垃圾回收语言通过定期检查哪些先前分配的内存仍然可以从应用程序的其他部分“访问”来帮助开发人员管理内存。垃圾回收语言中泄漏的主要原因是不需要的引用。如果你的 JavaScript 应用程序经常发生崩溃、高延迟和性能差,那么一个潜在的原因可能是内存泄漏。

在 JavaScript 中,内存是有生命周期的:

怎么避免JavaScript内存泄漏

  • 分配内存: 内存由操作系统分配,允许程序使用它。在 JavaScript 中,分配内存是自动完成的。

  • 使用内存: 这是程序实际使用先前分配的内存的空间。当在代码中使用分配的变量时,会发生读取和写入操作。

  • 释放内存: 释放不需要的内存,这样内存就会空闲并可以再次利用。在 JavaScript 中,释放内存是自动完成的。

在JavaScript中,对象会保存在堆内存中,可以根据引用链从根访问它们。垃圾收集器是 JavaScript 引擎中的一个后台进程,用于识别无法访问的对象、删除它们并回收内存。

下面是垃圾收集器根到对象的引用链示例:

怎么避免JavaScript内存泄漏

当内存中应该在垃圾回收周期中清理的对象,通过另一个对象的无意引用从根保持可访问时,就会发生内存泄漏。将冗余对象保留在内存中会导致应用程序内部使用过多的内存,并可能导致性能下降。

那该如何判断代码正在泄漏内存呢?通常,内存泄漏是很难被发现的,并且浏览器在运行它时不会抛出任何错误。如果注意到页面的性能越来越差,浏览器的内置工具可以帮助我们确定是否存在内存泄漏以及导致内存泄漏的对象。

内存使用检查最快的方法就是查看浏览器的任务管理器。 它们提供了当前在浏览器中运行的所有选项卡和进程的概览。在任务管理器中查看每个选项卡的 JavaScript 内存占用情况。如果网站什么都不做,但是 JavaScript 内存使用量却在逐渐增加,那么很有可能发生了内存泄漏。

怎么避免JavaScript内存泄漏

二、常见的内存泄漏

我们可以通过了解在 JavaScript 中如何创建不需要的引用来防止内存泄漏。以下情况就会导致不需要的引用。

1、意外的全局变量

全局变量始终可以从全局对象(在浏览器中,全局对象是window)中获得,并且永远不会被垃圾回收。在非严格模式下,以下行为会导致变量从局部范围泄露到全局范围:

(1)为未声明的变量赋值

这里我们给函数中一个未声明的变量bar赋值,这时就会使bar成为一个全局变量:

function foo(arg) {    bar = "hello world";}

这就等价于:

function foo(arg) {    window.bar = "hello world";}

这样就会创建一个多余的全局变量,当执行完foo函数之后,变量bar仍然会存在于全局对象中:

foo()window.bar   // hello world

(2)使用指向全局对象的 this

使用以下方式也会创建一个以外的全局变量:

function foo() {    this.bar = "hello world";}foo();

这里foo是在全局对象中调用的,所以其this是指向全局对象的(这里是window):

window.bar   // hello world

我们可以通过使用严格模式“use strict”来避免这一切。在JavaScript文件的开头,它将开启更严格的JavaScript解析模式,从而防止意外的创建全局变量。

需要特别注意那些用于临时存储和处理大量信息的全局变量。如果必须使用全局变量存储数据,就使用全局变量存储数据,但在不再使用时,就手动将其设置为 null,或者在处理完后重新分配。否则的话,请尽可能的使用局部变量。

2、 计时器

使用 setTimeout 或 setInterval 引用回调中的某个对象是防止对象被垃圾收集的最常见方法。如果我们在代码中设置了循环计时器,只要回调是可调用的,计时器回调中对对象的引用就会保持活动状态。

在下面的示例中,只有在清除计时器后,才能对数据对象进行垃圾收集。由于我们没有对setInterval的引用,所以它永远无法被清除和删除数据。hugeString会一直保存在内存中,直到应用程序停止,尽管从未使用过。

function setCallback() {  const data = {    counter: 0,    hugeString: new Array(100000).join('x')  };  return function cb() {    data.counter++;   // data对象是回调范围的一部分    console.log(data.counter);  }}setInterval(setCallback(), 1000);

当执行这段代码时,就会每秒输出一个数字:

怎么避免JavaScript内存泄漏

那我们如何去阻止他呢?尤其是在回调的寿命未定义或不确定的情况下:

  • 修改计时器回调中引用的对象;

  • 必要时使用从计时器返回的句柄(定时器的标识符)取消它。

function setCallback() {  // 将数据对象解包  let counter = 0;  const hugeString = new Array(100000).join('x'); // 在setCallback返回时被删除    return function cb() {    counter++; // 只有计数器counter是回调范围的一部分    console.log(counter);  }}const timerId = setInterval(setCallback(), 1000); // 保存定时器的ID// 合适的时机清除定时器clearInterval(timerId);

3、 闭包

我们知道,函数范围内的变量在函数退出调用堆栈后,如果函数外部没有任何指向它们的引用,则会被清除。尽管函数已经完成执行,其执行上下文和变量环境早已消失,但闭包将保持变量的引用和活动状态。

function outer() {  const potentiallyHugeArray = [];  return function inner() {    potentiallyHugeArray.push('Hello');      console.log('Hello');  };};const sayHello = outer();function repeat(fn, num) {  for (let i = 0; i < num; i++){    fn();  }}repeat(sayHello, 10);

显而易见,这里就形成了一个闭包。其输出结果如下:

怎么避免JavaScript内存泄漏

这里,potentiallyHugeArray 永远不会从任何函数返回,也无法访问,但它的大小可能会无限增长,这取决于调用函数 inner() 的次数。

那该如何防止这个问题呢?闭包是不可避免的,也是JavaScript不可或缺的一部分,因此重要的是:

  • 了解何时创建闭包以及闭包保留了哪些对象。

  • 了解闭包的预期寿命和用法(尤其是用作回调时)。

4、 事件监听器

活动事件侦听器将防止在其范围内捕获的所有变量被垃圾收集。添加后,事件侦听器将一直有效,直到:

  • 使用 removeEventListener() 显式删除。

  • 关联的 DOM 元素被移除。

对于某些类型的事件,它会一直保留到用户离开页面,就像应该多次单击的按钮一样。但是,有时我们希望事件侦听器执行一定次数。

const hugeString = new Array(100000).join('x');document.addEventListener('keyup', function() { // 匿名内联函数,无法删除它  doSomething(hugeString); // hugeString 将永远保留在回调的范围内});

在上面的示例中,匿名内联函数用作事件侦听器,这意味着不能使用 removeEventListener() 删除它。同样,document 不能被删除,因此只能使用 listener 函数以及它在其范围内保留的内容,即使只需要启动一次。

那该如何防止这个问题呢?一旦不再需要,我们应该通过创建指向事件侦听器的引用并将其传递给 removeEventListener() 来注销事件侦听器。

function listener() {  doSomething(hugeString);}document.addEventListener('keyup', listener); document.removeEventListener('keyup', listener);

如果事件侦听器只能执行一次,addEventListener() 可以接受第三个参数,这是一个提供附加选项的对象。假定将 {once:true} 作为第三个参数传递给 addEventListener() ,则侦听器函数将在处理一次事件后自动删除。

document.addEventListener('keyup', function listener() {  doSomething(hugeString);}, {once: true});

5、缓存

如果我们不断地将内存添加到缓存中,而不删除未使用的对象,并且没有一些限制大小的逻辑,那么缓存可以无限增长。

let user_1 = { name: "Peter", id: 12345 };let user_2 = { name: "Mark", id: 54321 };const mapCache = new Map();function cache(obj){  if (!mapCache.has(obj)){    const value = `${obj.name} has an id of ${obj.id}`;    mapCache.set(obj, value);    return [value, 'computed'];  }  return [mapCache.get(obj), 'cached'];}cache(user_1); // ['Peter has an id of 12345', 'computed']cache(user_1); // ['Peter has an id of 12345', 'cached']cache(user_2); // ['Mark has an id of 54321', 'computed']console.log(mapCache); // {{…} => 'Peter has an id of 12345', {…} => 'Mark has an id of 54321'}user_1 = null;console.log(mapCache); // {{…} => 'Peter has an id of 12345', {…} => 'Mark has an id of 54321'}

在上面的示例中,缓存仍然保留 user_1 对象。因此,我们需要将那些永远不会被重用的变量从缓存中清除。

可以使用 WeakMap 来解决此问题。它是一种具有弱键引用的数据结构,仅接受对象作为键。如果我们使用一个对象作为键,并且它是对该对象的唯一引用&mdash;&mdash;相关变量将从缓存中删除并被垃圾收集。在以下示例中,将 user_1 对象清空后,相关变量会在下一次垃圾回收后自动从 WeakMap 中删除。

let user_1 = { name: "Peter", id: 12345 };let user_2 = { name: "Mark", id: 54321 };const weakMapCache = new WeakMap();function cache(obj){  // ...  return [weakMapCache.get(obj), 'cached'];}cache(user_1); // ['Peter has an id of 12345', 'computed']cache(user_2); // ['Mark has an id of 54321', 'computed']console.log(weakMapCache); // {(…) => "Peter has an id of 12345", (…) => "Mark has an id of 54321"}user_1 = null;console.log(weakMapCache); // {(…) => "Mark has an id of 54321"}

6、分离的DOM元素

如果DOM节点具有来自 JavaScript 的直接引用,它将防止对其进行垃圾收集,即使在从DOM树中删除该节点之后也是如此。

在下面的示例中,创建了一个div元素并将其附加到 document.body 中。removeChild() 就无法按预期工作,堆快照将显示分离的htmlDivElement,因为仍有一个变量指向div。

function createElement() {  const div = document.createElement('div');  div.id = 'detached';  return div;}// 即使在调用deleteElement() 之后,它仍将继续引用DOM元素const detachedDiv = createElement();document.body.appendChild(detachedDiv);function deleteElement() {  document.body.removeChild(document.getElementById('detached'));}deleteElement();

要解决此问题,可以将DOM引用移动到本地范围。在下面的示例中,在函数appendElement() 完成后,将删除指向DOM元素的变量。

function createElement() {...}// DOM引用在函数范围内function appendElement() {  const detachedDiv = createElement();  document.body.appendChild(detachedDiv);}appendElement();function deleteElement() {  document.body.removeChild(document.getElementById('detached'));}deleteElement();

三、识别内存泄漏

调试内存问题是一项复杂的工作,我们可以使用 Chrome DevTools 来识别内存图和一些内存泄漏,我们需要关注以下两个方面:

  • 使用性能分析器可视化内存消耗。

  • 识别分离的 DOM 节点。

1、使用性能分析器可视化内存消耗

以下面的代码为例,有两个按钮:打印和清除。点击“打印”按钮,通过创建 paragraph 节点并将大字符串设置到全局,将1到10000的数字追加到DOM中。

“清除”按钮会清除全局变量并覆盖 body 的正文,但不会删除单击“打印”时创建的节点:

<!DOCTYPE html><html lang="en">  <head>    <title>Memory leaks</title>  </head>  <body>    <button id="print">打印</button>    <button id="clear">清除</button>  </body></html><script>  var longArray = [];    function print() {    for (var i = 0; i < 10000; i++) {      let paragraph = document.createElement("p");      paragraph.innerHTML = i;      document.body.appendChild(paragraph);    }    longArray.push(new Array(1000000).join("y"));  }    document.getElementById("print").addEventListener("click", print);  document.getElementById("clear").addEventListener("click", () => {    window.longArray = null;    document.body.innerHTML = "Cleared";  });</script>

当每次点击打印按钮时,JavaScript Heap都会出现蓝色的峰值,并逐渐增加,这是因为JavaScript正在创建DOM节点并字符串添加到全局数组。当点击清除按钮时,JavaScript Heap就变得正常了。除此之外,可以看到节点的数量(绿色的线)一直在增加,因为我们并没有删除这些节点。

怎么避免JavaScript内存泄漏

在实际的场景中,如果观察到内存持续出现峰值,并且内存消耗一直没有减少,那可能存在内存泄露。

2、 识别分离的 DOM 节点

当一个节点从 DOM 树中移除时,它被称为分离,但一些 JavaScript 代码仍然在引用它。让我们使用下面的代码片段检查分离的 DOM 节点。通过单击按钮,可以将列表元素添加到其父级中并将父级分配给全局变量。简单来说,全局变量保存着 DOM 引用:

var detachedElement;function createList(){  let ul = document.createElement("ul");  for(let i = 0; i < 5; i++){    ul.appendChild(document.createElement("li"));  }  detachedElement = ul;}document.getElementById("createList").addEventListener("click", createList);

我们可以使用 heap snapshot 来检查分离的DOM节点,可以在Chrome DevTools 的Memory面板中打开Heap snapshots选项:

怎么避免JavaScript内存泄漏

点击页面的按钮后,点击下面蓝色的Take snapshot按钮,我们可以在中间的搜索栏目输入Detached来过滤结果以找到分离的DOM节点,如下所示:

怎么避免JavaScript内存泄漏

当然也可以尝试使用此方法来识别其他内存泄漏。

关于“怎么避免JavaScript内存泄漏”的内容就介绍到这里了,感谢大家的阅读。如果想了解更多行业相关的知识,可以关注编程网精选频道,小编每天都会为大家更新不同的知识点。

--结束END--

本文标题: 怎么避免JavaScript内存泄漏

本文链接: https://www.lsjlt.com/news/330265.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • 怎么避免JavaScript内存泄漏
    这篇文章主要介绍“怎么避免JavaScript内存泄漏”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“怎么避免JavaScript内存泄漏”文章能帮助大家解决问题。一、什么是内存泄漏JavaScrip...
    99+
    2023-06-30
  • 一文搞懂如何避免JavaScript内存泄漏
    目录一、什么是内存泄漏二、常见的内存泄漏1、意外的全局变量2、 计时器3、 闭包4、 事件监听器5、缓存6、分离的DOM元素三、识别内存泄漏1、使用性能分析器可视化内存消耗2、 识别...
    99+
    2022-11-13
  • C#开发中如何避免内存泄漏
    C#开发中如何避免内存泄漏,需要具体代码示例内存泄漏是软件开发过程中常见的问题之一,特别是在使用C#语言进行开发时。内存泄漏会导致应用程序占用越来越多的内存空间,最终导致程序运行缓慢甚至崩溃。为了避免内存泄漏,我们需要注意一些常见的问题并采...
    99+
    2023-10-22
    垃圾回收 内存管理 资源释放
  • JavaScript函数中的Java数组:如何避免内存泄漏?
    在使用JavaScript函数时,经常需要使用Java数组。然而,这种用法有时会导致内存泄漏,从而影响应用程序的性能和稳定性。那么,如何避免这种情况呢?本文将介绍几种有效的方法来解决这个问题。 使用数组的静态变量 当定义一个Java数...
    99+
    2023-10-25
    数组 javascript 函数
  • JavaScript中内存泄漏怎么办
    这篇文章主要介绍JavaScript中内存泄漏怎么办,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!一、什么是内存泄漏?程序的运行需要内存。只要程序提出要求,操作系统或者运行时(run...
    99+
    2022-10-19
  • 怎么排查Javascript内存泄漏
    这篇文章主要讲解了“怎么排查Javascript内存泄漏”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“怎么排查Javascript内存泄漏”吧!如何判断我的应用发生了内存泄漏为了证明螃蟹的听...
    99+
    2023-07-02
  • JavaScript中怎么防范内存泄漏
    这篇文章给大家介绍JavaScript中怎么防范内存泄漏,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。浏览器将对象保留在堆内存中,通过引用链可从根对象到达这些对象。垃圾回收器(GC)是...
    99+
    2022-10-19
  • javascript中怎么防止内存泄漏
    小编给大家分享一下javascript中怎么防止内存泄漏,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧! ...
    99+
    2022-10-19
  • Python 中的文件缓存对象:如何避免内存泄漏?
    Python 是一门功能强大的编程语言,广泛应用于各种领域,包括数据分析、机器学习、Web 开发等等。Python 中的文件缓存对象是一个非常重要的概念,可以帮助我们提高程序的性能和效率。但是,如果使用不当,会导致内存泄漏的问题。本文将介绍...
    99+
    2023-07-11
    文件 缓存 对象
  • NumPy 打包 Python 编程算法:如何避免内存泄漏?
    Python 是一门动态解释型语言,因其简单易学、灵活、高效而成为数据科学领域最流行的编程语言之一。NumPy 是 Python 中用于科学计算的核心库之一,提供了多维数组对象、各种派生对象(如掩码数组和矩阵)以及用于数组操作的函数和方法...
    99+
    2023-06-30
    打包 numy 编程算法
  • 避免JavaScript内存泄露的方法有哪些
    本篇内容主要讲解“避免JavaScript内存泄露的方法有哪些”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“避免JavaScript内存泄露的方法有哪些”吧!简...
    99+
    2022-10-19
  • JavaScript内存泄漏和内存溢出是什么
    本篇内容主要讲解“JavaScript内存泄漏和内存溢出是什么”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“JavaScript内存泄漏和内存溢出是什么”吧! ...
    99+
    2022-10-19
  • android中用getApplicationContext()会不会避免某些内存泄漏问题?
    作者:Hewi 链接:https://www.zhihu.com/question/34007989/answer/58296467 来源:知乎 著作权归作者所有。商业转载请联...
    99+
    2022-06-06
    内存泄漏 Android
  • ASP编程中,如何避免容器的内存泄漏问题?
    在ASP编程中,我们经常使用容器来存储数据,如数组、哈希表、列表等。这些容器在使用过程中,可能会出现内存泄漏的问题,导致程序运行变慢、内存占用过高等问题。本文将介绍如何避免容器的内存泄漏问题,并提供演示代码。 一、什么是内存泄漏? 内存泄...
    99+
    2023-06-01
    leetcode 编程算法 容器
  • JavaScript内存泄漏实例分析
    这篇文章主要讲解了“JavaScript内存泄漏实例分析”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“JavaScript内存泄漏实例分析”吧!js 内存泄...
    99+
    2022-10-19
  • JavaScript中内存泄漏指的是什么
    内存泄漏是指程序中一个对象被分配到内存中既不能使用,又不能回收,留在了堆内存中就称为内存泄漏,当一个对象已经不需要再使用本该被回收时,另外一个正在使用的对象持有它的引用从而导致它不能被回收,这导致本该被回收的对象不能被回收而停留在堆内存中,...
    99+
    2022-10-15
  • golang内存泄漏怎么排查
    在 Go 语言中,内存泄漏通常是由于不正确地使用或管理指针和引用导致的。以下是一些排查内存泄漏的常用方法:1. 使用 go buil...
    99+
    2023-10-21
    golang
  • ThreadLocal内存泄漏怎么预防
    这篇文章主要介绍“ThreadLocal内存泄漏怎么预防”,在日常操作中,相信很多人在ThreadLocal内存泄漏怎么预防问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”ThreadLocal内存泄漏怎么预防...
    99+
    2023-06-29
  • java怎么检查内存泄漏
    本篇内容介绍了“java怎么检查内存泄漏”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!内存泄漏场景长生命周期的对象持有短生命周期对象的引用就...
    99+
    2023-06-30
  • java内存泄漏怎么排查
    Java内存泄漏是指在程序运行过程中,不再使用的对象仍然占用着内存空间,导致内存无法被回收。以下是一些常见的排查内存泄漏的方法:1....
    99+
    2023-08-31
    java
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作