【转】JavaScript 中的垃圾回收和内存泄露如何处理
转自 https://blog.csdn.net/csdnnews/article/details/89596750?ops_request_misc=%257B%2522request%255Fid%2522%253A%2522159411067719195239861095%2522%252C%2522scm%2522%253A%252220140713.130102334…%2522%257D&request_id=159411067719195239861095&biz_id=0&utm_medium=distribute.pc_search_result.none-task-blog-2allfirst_rank_ecpm_v3~pc_rank_v3-1-89596750.pc_ecpm_v3_pc_rank_v3&utm_term=JavaScript+%E4%B8%AD%E7%9A%84%E5%9E%83%E5%9C%BE%E5%9B%9E%E6%94%B6%E5%92%8C%E5%86%85%E5%AD%98%E6%B3%84%E9%9C%B2%E5%A6%82%E4%BD%95%E5%A4%84%E7%90%86
作者 | 浪里行舟
责编 | 郭芮
程序的运行需要内存。只要程序提出要求,操作系统或者运行时就必须供给内存。所谓的内存泄漏简单来说是不再用到的内存,没有及时释放。为了更好避免内存泄漏,我们先介绍Javascript垃圾回收机制。
在C与C++等语言中,开发人员可以直接控制内存的申请和回收。但是在Java、C#、JavaScript语言中,变量的内存空间的申请和释放都由程序自己处理,开发人员不需要关心。也就是说,Javascript具有自动垃圾回收机制(Garbage Collecation)。
垃圾回收的必要性
下面这段话引自《JavaScript权威指南(第四版)》:
由于字符串、对象和数组没有固定大小,所有当他们的大小已知时,才能对他们进行动态的存储分配。JavaScript程序每次创建字符串、数组或对象时,解释器都必须分配内存来存储那个实体。只要像这样动态地分配了内存,最终都要释放这些内存以便他们能够被再用,否则,JavaScript的解释器将会消耗完系统中所有可用的内存,造成系统崩溃。
这段话解释了为什么需要系统需要垃圾回收,JavaScript不像C/C++,它有自己的一套垃圾回收机制。
JavaScript垃圾回收的机制很简单:找出不再使用的变量,然后释放掉其占用的内存,但是这个过程不是时时的,因为其开销比较大,所以垃圾回收器会按照固定的时间间隔周期性的执行。
- var a = "浪里行舟";
- var b = "前端工匠";
- var a = b; //重写a
这段代码运行之后,“浪里行舟”这个字符串失去了引用(之前是被a引用),系统检测到这个事实之后,就会释放该字符串的存储空间以便这些空间可以被再利用。
垃圾回收机制
垃圾回收机制怎么知道,哪些内存不再需要呢?
垃圾回收有两种方法:标记清除、引用计数。引用计数不太常用,标记清除较为常用。
1.标记清除
这是javascript中最常用的垃圾回收方式。当变量进入执行环境是,就标记这个变量为“进入环境”。从逻辑上讲,永远不能释放进入环境的变量所占用的内存,因为只要执行流进入相应的环境,就可能会用到他们。当变量离开环境时,则将其标记为“离开环境”。
垃圾收集器在运行的时候会给存储在内存中的所有变量都加上标记。然后,它会去掉环境中的变量以及被环境中的变量引用的标记。而在此之后再被加上标记的变量将被视为准备删除的变量,原因是环境中的变量已经无法访问到这些变量了。最后。垃圾收集器完成内存清除工作,销毁那些带标记的值,并回收他们所占用的内存空间。
我们用个例子,解释下这个方法:
- var m = 0,n = 19 // 把 m,n,add() 标记为进入环境。
- add(m, n) // 把 a, b, c标记为进入环境。
- console.log(n) // a,b,c标记为离开环境,等待垃圾回收。
- function add(a, b) {
- a++
- var c = a + b
- return c
- }
2.引用计数
所谓"引用计数"是指语言引擎有一张"引用表",保存了内存里面所有的资源(通常是各种值)的引用次数。如果一个值的引用次数是0,就表示这个值不再用到了,因此可以将这块内存释放。
上图中,左下角的两个值,没有任何引用,所以可以释放。
如果一个值不再需要了,引用数却不为0,垃圾回收机制无法释放这块内存,从而导致内存泄漏。
- var arr = [1, 2, 3, 4];
- arr = [2, 4, 5]
- console.log('浪里行舟');
上面代码中,数组[1, 2, 3, 4]是一个值,会占用内存。变量arr是仅有的对这个值的引用,因此引用次数为1。尽管后面的代码没有用到arr,它还是会持续占用内存。至于如何释放内存,我们下文介绍。
第三行代码中,数组[1, 2, 3, 4]引用的变量arr又取得了另外一个值,则数组[1, 2, 3, 4]的引用次数就减1,此时它引用次数变成0,则说明没有办法再访问这个值了,因而就可以将其所占的内存空间给收回来。
但是引用计数有个最大的问题:循环引用
- function func() {
- let obj1 = {};
- let obj2 = {};
- obj1.a = obj2; // obj1 引用 obj2
- obj2.a = obj1; // obj2 引用 obj1
- }
当函数 func 执行结束后,返回值为 undefined,所以整个函数以及内部的变量都应该被回收,但根据引用计数方法,obj1 和 obj2 的引用次数都不为 0,所以他们不会被回收。
要解决循环引用的问题,最好是在不使用它们的时候手工将它们设为空。上面的例子可以这么做:
- obj1 = null;
- obj2 = null;
哪些情况会引起内存泄漏?
虽然JavaScript会自动垃圾收集,但是如果我们的代码写法不当,会让变量一直处于“进入环境”的状态,无法被回收。
下面列一下内存泄漏常见的几种情况:
1.意外的全局变量
- function foo(arg) {
- bar = "this is a hidden global variable";
- }
bar没被声明,会变成一个全局变量,在页面关闭之前不会被释放。
另一种意外的全局变量可能由 this 创建:
- function foo() {
- this.variable = "potential accidental global";
- }
- // foo 调用自己,this 指向了全局对象(window)
- foo();
在 JavaScript 文件头部加上 'use strict',可以避免此类错误发生。启用严格模式解析 JavaScript ,避免意外的全局变量。
2.被遗忘的计时器或回调函数
- var someResource = getData();
- setInterval(function() {
- var node = document.getElementById('Node');
- if(node) {
- // 处理 node 和 someResource
- node.innerHTML = JSON.stringify(someResource));
- }
- }, 1000);
这样的代码很常见,如果id为Node的元素从DOM中移除,该定时器仍会存在,同时,因为回调函数中包含对someResource的引用,定时器外面的someResource也不会被释放。
3.闭包
- function bindEvent(){
- var obj=document.createElement('xxx')
- obj.οnclick=function(){
- // Even if it is a empty function
- }
- }
闭包可以维持函数内局部变量,使其得不到释放。上例定义事件回调时,由于是函数内定义函数,并且内部函数--事件回调引用外部函数,形成了闭包。
- // 将事件处理函数定义在外面
- function bindEvent() {
- var obj = document.createElement('xxx')
- obj.onclick = onclickHandler
- }
- // 或者在定义事件处理函数的外部函数中,删除对dom的引用
- function bindEvent() {
- var obj = document.createElement('xxx')
- obj.onclick = function() {
- // Even if it is a empty function
- }
- obj = null
- }
解决之道,将事件处理函数定义在外部,解除闭包,或者在定义事件处理函数的外部函数中,删除对dom的引用。
4.没有清理的DOM元素引用
有时,保存 DOM 节点内部数据结构很有用。假如你想快速更新表格的几行内容,把每一行 DOM 存成字典(JSON 键值对)或者数组很有意义。此时,同样的 DOM 元素存在两个引用:一个在 DOM 树中,另一个在字典中。将来你决定删除这些行时,需要把两个引用都清除。
- var elements = {
- button: document.getElementById('button'),
- image: document.getElementById('image'),
- text: document.getElementById('text')
- };
- function doStuff() {
- image.src = 'http://some.url/image';
- button.click();
- console.log(text.innerHTML);
- }
- function removeButton() {
- document.body.removeChild(document.getElementById('button'));
- // 此时,仍旧存在一个全局的 #button 的引用
- // elements 字典。button 元素仍旧在内存中,不能被 GC 回收。
- }
虽然我们用removeChild移除了button,但是还在elements对象里保存着#button的引用,换言之,DOM元素还在内存里面。
内存泄漏的识别方法
新版本的chrome在 performance 中查看:
步骤:
打开开发者工具 Performance;
勾选 Screenshots 和 memory;
左上角小圆点开始录制(record);
停止录制。
图中 Heap 对应的部分就可以看到内存在周期性的回落也可以看到垃圾回收的周期,如果垃圾回收之后的最低值(我们称为min),min在不断上涨,那么肯定是有较为严重的内存泄漏问题。
避免内存泄漏的一些方式:
减少不必要的全局变量,或者生命周期较长的对象,及时对无用的数据进行垃圾回收;
注意程序逻辑,避免“死循环”之类的;
避免创建过多的对象。
总而言之需要遵循一条原则:不用了的东西要及时归还。
垃圾回收的使用场景优化
1.数组array优化
将[]赋值给一个数组对象,是清空数组的捷径(例如:arr = [];),但是需要注意的是,这种方式又创建了一个新的空对象,并且将原来的数组对象变成了一小片内存垃圾!实际上,将数组长度赋值为0(arr.length = 0)也能达到清空数组的目的,并且同时能实现数组重用,减少内存垃圾的产生。
- const arr = [1, 2, 3, 4];
- console.log('浪里行舟');
- arr.length = 0 // 可以直接让数字清空,而且数组类型不变。
- // arr = []; 虽然让a变量成一个空数组,但是在堆上重新申请了一个空数组对象。
2. 对象尽量复用
对象尽量复用,尤其是在循环等地方出现创建新对象,能复用就复用。不用的对象,尽可能设置为null,尽快被垃圾回收掉。
- var t = {} // 每次循环都会创建一个新对象。
- for (var i = 0; i < 10; i++) {
- // var t = {};// 每次循环都会创建一个新对象。
- t.age = 19
- t.name = '123'
- t.index = i
- console.log(t)
- }
- t = null //对象如果已经不用了,那就立即设置为null;等待垃圾回收。
3.在循环中的函数表达式,能复用最好放到循环外面
- // 在循环中最好也别使用函数表达式。
- for (var k = 0; k < 10; k++) {
- var t = function(a) {
- // 创建了10次 函数对象。
- console.log(a)
- }
- t(k)
- }
- // 推荐用法
- function t(a) {
- console.log(a)
- }
- for (var k = 0; k < 10; k++) {
- t(k)
- }
- t = null
Python 第一地位稳了!微软正式拥抱 Python !
https://edu.csdn.net/topic/python115?utm_source=csdn_bw
作者:浪里行舟,硕士研究生,专注于前端,运营有个人公众号前端工匠,致力于打造适合初中级工程师能够快速吸收的一系列优质文章。
声明:本文为CSDN技术头条专栏原创投稿,未经允许请勿转载。
作为码一代,想教码二代却无从下手:
听说少儿编程很火,可它有哪些好处呢?
孩子多大开始学习比较好呢?又该如何学习呢?
最新的编程教育政策又有哪些呢?
下面给大家介绍CSDN新成员:极客宝宝(ID:geek_baby)
戳他了解更多↓↓↓
热 文 推 荐
☞ 北大毕业名企 CTO 为何坐在泰国海边敲代码? | 人物志
☞淘宝和拼多多争夺 50 万亿电商江湖 | 畅言
☞取代不了 C/C++ 的 Rust 如何“逆袭”?
☞ 国际信奥金牌,保送清华姚班,这位 00 后是怎么做到的?| 人物志
程序员的双肩包,大概能装下整个宇宙!
☞ 强推!十大顶级大数据可视化工具 | 程序员硬核评测
裁员25%, 梅西也拯救不了全球第一款区块链手机!
深入卷积神经网络背后的数学原理 | 技术头条
☞ 程序员被骗"黑砖窑":监禁、恐吓、996无休编程!
System.out.println("点个在看吧!");
console.log("点个在看吧!");
print("点个在看吧!");
printf("点个在看吧!\n");
cout << "点个在看吧!" << endl;
Console.WriteLine("点个在看吧!");
Response.Write("点个在看吧!");
alert("点个在看吧!")
echo "点个在看吧!"
点击阅读原文,输入关键词,即可搜索您想要的 CSDN 文章。
你点的每个“在看”,我都认真当成了喜欢
【转】JavaScript 中的垃圾回收和内存泄露如何处理相关推荐
- JavaScript 中的垃圾回收和内存泄露如何处理?| 技术头条
作者 | 浪里行舟 责编 | 郭芮 程序的运行需要内存.只要程序提出要求,操作系统或者运行时就必须供给内存.所谓的内存泄漏简单来说是不再用到的内存,没有及时释放.为了更好避免内存泄漏,我们先介绍Jav ...
- JavaScript中的垃圾回收和内存泄漏
前言 程序的运行需要内存.只要程序提出要求,操作系统或者运行时就必须供给内存.所谓的内存泄漏简单来说是不再用到的内存,没有及时释放.为了更好避免内存泄漏,我们先介绍Javascript垃圾回收机制. ...
- 【java虚拟机序列】java中的垃圾回收与内存分配策略
在[java虚拟机系列]java虚拟机系列之JVM总述中我们已经详细讲解过java中的内存模型,了解了关于JVM中内存管理的基本知识,接下来本博客将带领大家了解java中的垃圾回收与内存分配策略. 垃 ...
- 浏览器中的垃圾回收与内存泄漏
1. 介绍 浏览器的 Javascript 具有自动垃圾回收机制(GC:Garbage Collecation),也就是说,执行环境会负责管理代码执行过程中使用的内存.其原理是:垃圾收集器会定期(周期 ...
- 闭包,作用域链,垃圾回收,内存泄露
关于闭包,我翻了几遍书,看了几遍视频,查了一些资料,可是还是迷迷糊糊的,干脆自己动手来个总结吧 !欢迎指正... (- o -)~zZ 1. 什么是闭包? 来看一些关于闭包的定义: 闭包是指有权访问另 ...
- js的垃圾回收和内存泄露
垃圾回收机制 浏览器的 Javascript 具有自动垃圾回收机制(GC:Garbage Collecation),也就是说,执行环境会负责管理代码执行过程中使用的内存.其原理是:垃圾收集器会定期(周 ...
- JavaScript 的垃圾回收与内存泄露
2019独角兽企业重金招聘Python工程师标准>>> JavaScript采用垃圾自动回收机制,运行时环境会自动清理不再使用的内存,因此javascript无需像C++等语言一样手 ...
- flash垃圾回收和内存泄露
GC的原子模型 之所以用"原子模型"(Atomic Model)这个词,是因为这篇文章是用来描述GC怎么样在 player中工作的模型,但是不包涵其相关的技术实现或细节描述. 之所 ...
- Python垃圾回收与内存泄露
Python是面向对象.高级编程语言,其世界里万物皆对象,当我们编写的程序运行时,代码中定义的对象在物理内存中会占用相应的空间.现在流行的高级语言如Java,C#等都采用了垃圾收集机制自动管理内存使用 ...
最新文章
- c语言i o编程,C 语言输入输出 (I/O)
- 基于 OSGi 的面向服务的组件编程
- 科大星云诗社动态20210827
- 网络安全-windowserver搭建DHCP服务器
- 一台电脑同时运行多个tomcat配置方法
- ERROR 1045 (28000): Access denied for user 'ODBC'@'localhost' 解决Mysql错误
- tf 矩阵行和列交换_TF-搞不懂的TF矩阵加法
- 工作总结20:遮挡层
- c语言中foreach的用法,详解JavaScript中的forEach()方法的使用
- Running Trinity in multiple steps
- 【华为云技术分享】Batch Normalization (BN) 介绍
- java 大数操作(加减乘除及取余)
- 博为峰Java技术文章 ——JavaSE Swing JInternalFrame内部窗体面板II
- 2021-08-08 WPF控件专题 StackPanel 控件详解
- LabVIEW强制重新安装无法运行或损坏的NI软件
- 有关GUASS高斯数据库的语法汇总(获取字符串字节数等)
- 计算机网络数据爆分片MTU,踢走绊脚石,MTU解析与常见问题汇总-上篇
- 菜花这样做,就着它吃光三碗饭
- miktex报错:the remote package repository is outdated
- 错误: Unable to find explicit activity class ...have you declared this activity in your AndroidMa
热门文章
- Spark MLlib 入门--Breeze函数
- 数据可视化实战——线条、颜色、背景、字体、标记、图例的设置与基本可视化图形的绘制
- kafka修改服务器时间,Kafka笔记2--Kafka的服务端配置
- Tacotron: Towards End-to-End Speech Synthesis
- 码农二代有多强?学区房?不存在的!
- 树莓派教程(基于python编程)--初级篇
- 郑大计算机专业保研都去了哪些大学,郑州大学一个班30人全部保研 过半保送“985”和中科院!...
- 企业能把员工转为劳务派遣工吗
- 关于《流浪地球》炸裂的口碑,机器学习竟然是这样评价的————Python文本情感极性分析详解(上)
- Boosting算法总结