Javascript 垃圾收集机制介绍理解


Posted in Javascript onMay 14, 2013

经常使用 Javascript 的人会琢磨其垃圾收集机制,Javascript 并不像 C,C++ 那样需要开发者手动去清除垃圾,在编写 Javascript 程序是,开发者无需关心内存使用问题,所需内存分配以及无用内存(垃圾)的回收完全实现了自动管理。究其根源,主要是程序收集那些不再使用的变量,并且释放其占用的内存。因此,垃圾收集机制会按照固定时间间隔,周期性反复的执行这一操作。

举例来说,局部变量只存在于函数内部,程序会为局部变量在栈内存或堆内存中分配对应的存储空间,当函数运行结束,局部变量所占用的内存就没有存在的必要了,这时程序会释放局部变量所占用的内存供其他变量使用。这是程序最简单释放内存的方法,但是很多时候,程序中变量会一直被使用,此时垃圾收集机制必须跟踪变量并且判断其是否被使用,是否可以释放其内存空间。

垃圾收集机制主要判断变量释放内存空间的方法有两个:其一是标记清除法,其二是引用计数法。

标记法,每个变量都有其运行环境,变量创建后会在某种环境中运行,比如创建一个局部变量,局部变量会运行在函数体内。当函数运行时,会标记局部变量为“进入环境”,当函数体运行结束后,意味着变量脱离了其运行环境,此时则将变量标记为“离开环境”。对于“离开环境”的变量,垃圾收集机制会进行相应记录,并且在下一个回收周期时将其释放。

引用计数法,跟踪记录每个值的被引用次数。声明一个变量并将一个引用类型值赋给该变量时,这个值得引用次数就是 1。如果同一个值又被赋给另外一个变量,则该值的引用次数加 1。相反,如果包含对这个值的引用的变量又取得另外一个值,这个值得引用次数减 1。当这个值得引用次数为 0 时,则说明没有办法再访问到此值,因此就可以将其占用的内存空间回收。当垃圾收集器在下一个周期运行时,会释放引用次数为零的值所占用的内存空间。(原文解释参考:Javascript 高级程序设计 - 第二版)

举个例子来说:

            function countMethod(){
                  var object1 = new Object(); // 声明变量,计数器由 0 变为 1
                  var object2 = new Object(); // 声明变量,计数器由 0 变为 1
                  object1.method1 = object2;  // object1 计数器 -1,object2 计数器 +1
                  object2.method2 = object1;  // object1 计数器 +1,object2 计数器 -1
            }

此函数运行退出后,object1 的计数器读数为 1,object2 的计数器度数为 1。所以两个变量都不会被销毁。如果大量的这样的程序存在于函数体内,就会导致大量的内存被浪费而无法回收,从而导致内存的泄露。

上述问题解决方法,手动释放 object1 object2 所占用的内存。即:

                 object1.method1 = null;
                 object2.method2 = null;

对比上面的例子,举一个正常情况下的例子。
            function countMethod(){
                  var object1 = new Object(); // 声明变量,计数器由 0 变为 1
                  var object2 = new Object(); // 声明变量,计数器由 0 变为 1
                  object1.method1 = "This is object1";  // object1 计数器 -1,object1 读数变为0
                  object2.method2 = "This is object2";  // object2 计数器 -1,object2 读数变为0
            }

通过上例看出,正常情况下,当函数运行结束后,object1 object2的读数均为 0,在下一个垃圾收集周期时,会被回收并且释放其所占用的内存。
Javascript 相关文章推荐
优化javascript的执行效率一些方法总结
Dec 25 Javascript
打造个性化的功能强大的Jquery虚拟键盘(VirtualKeyboard)
Oct 11 Javascript
node.js中使用socket.io的方法
Dec 15 Javascript
bootstrap与Jquery UI 按钮样式冲突的解决办法
Sep 23 Javascript
浅谈node模块与npm包管理工具
Jan 03 Javascript
JavaScript之实现一个简单的Vue示例
Jan 17 Javascript
django中使用vue.js的要点总结
Jul 07 Javascript
如何基于JavaScript判断图片是否加载完成
Dec 28 Javascript
解决vue elementUI中table里数字、字母、中文混合排序问题
Jan 07 Javascript
微信小程序纯文本实现@功能
Apr 08 Javascript
vite+vue3.0+ts+element-plus快速搭建项目的实现
Jun 24 Vue.js
Vue.js中v-for指令的用法介绍
Mar 13 Vue.js
JavaScript实现GriwView单列全选(自写代码)
May 13 #Javascript
jquery实现漂浮在网页右侧的qq在线客服插件示例
May 13 #Javascript
js 程序执行与顺序实现详解
May 13 #Javascript
JS/jQuery实现默认显示部分文字点击按钮显示全部内容
May 13 #Javascript
JS 加入收藏夹的代码(主流浏览器通用)
May 13 #Javascript
jQuery实现长文字部分显示代码
May 13 #Javascript
jq选项卡鼠标延迟的插件实例
May 13 #Javascript
You might like
PHP 多维数组排序(usort,uasort)
2010/06/30 PHP
php获取淘宝分类id示例
2014/01/16 PHP
PHP中SimpleXML函数用法分析
2014/11/26 PHP
使用PHP生成PDF方法详解
2015/01/23 PHP
PHP的垃圾回收机制代码实例讲解
2021/02/27 PHP
可以支持多中格式的JS键盘
2007/05/02 Javascript
jQuery判断对象是否存在的方法
2015/02/05 Javascript
jquery表单对象属性过滤选择器实例分析
2015/05/18 Javascript
jQuery实现下滑菜单导航效果代码
2015/08/25 Javascript
谈谈JavaScript类型系统之Math
2016/01/06 Javascript
详解angularJs模块ui-router之状态嵌套和视图嵌套
2017/04/28 Javascript
vue.js国际化 vue-i18n插件的使用详解
2017/07/07 Javascript
nodeJS进程管理器pm2的使用
2019/01/09 NodeJs
vue多个元素的样式选择器问题
2019/11/29 Javascript
JavaScript 中的无穷数(Infinity)详解
2020/02/13 Javascript
PyCharm使用教程之搭建Python开发环境
2016/06/07 Python
Python入门_浅谈逻辑判断与运算符
2017/05/16 Python
Python 实现12306登录功能实例代码
2018/02/09 Python
python线程池threadpool使用篇
2018/04/27 Python
在python2.7中用numpy.reshape 对图像进行切割的方法
2018/12/05 Python
python encrypt 实现AES加密的实例详解
2020/02/20 Python
Python实现疫情通定时自动填写功能(附代码)
2020/05/27 Python
声乐专业大学生职业生涯规划书:理想的未来需要自己去打造
2014/09/20 职场文书
医院见习报告范文
2014/11/03 职场文书
个人先进事迹材料范文
2014/12/29 职场文书
介绍信怎么写
2015/01/30 职场文书
护士辞职信怎么写
2015/02/27 职场文书
车间质检员岗位职责
2015/04/08 职场文书
2015年健康教育工作总结
2015/04/10 职场文书
2016同学毕业寄语大全
2015/12/04 职场文书
加强党性修养心得体会
2016/01/21 职场文书
大学生村官驻村工作心得体会
2016/01/23 职场文书
导游词之苏州寒山寺
2019/12/05 职场文书
Feign调用传输文件异常的解决
2021/06/24 Java/Android
一文搞懂Golang 时间和日期相关函数
2021/12/06 Golang
Android实现图片九宫格
2022/06/28 Java/Android