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 相关文章推荐
js chrome浏览器判断代码
Mar 28 Javascript
jquery得到font-size属性值实现代码
Sep 30 Javascript
jquery实现手机发送验证码的倒计时代码
Feb 12 Javascript
javascript中的this详解
Dec 08 Javascript
JS对字符串编码的几种方式使用指南
May 14 Javascript
Bootstrap框架动态生成Web页面文章内目录的方法
May 12 Javascript
jQuery选择器_动力节点Java学院整理
Jul 05 jQuery
JavaScript实现全选取消效果
Dec 14 Javascript
Vue 中使用 CSS Modules优雅方法
Apr 09 Javascript
详解vuex commit保存数据技巧
Dec 25 Javascript
vue组件开发props验证的实现
Feb 12 Javascript
JS setTimeout与setInterval的区别
Apr 20 Javascript
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 ctype函数中文翻译和示例
2014/03/21 PHP
ThinkPHP中调用PHPExcel的实现代码
2017/04/08 PHP
php使用ftp实现文件上传与下载功能
2017/07/21 PHP
使用js检测浏览器的实现代码
2013/05/14 Javascript
jquery简单实现滚动条下拉DIV固定在头部不动
2013/11/25 Javascript
根据表格中的某一列进行排序的javascript代码
2013/11/29 Javascript
jquery访问ashx文件示例代码
2014/08/11 Javascript
上传图片js判断图片尺寸和格式兼容IE
2014/09/01 Javascript
Lab.js初次使用笔记
2015/02/28 Javascript
通过javascript进行UTF-8编码的实现方法
2016/06/27 Javascript
AngularJs bootstrap搭载前台框架——基础页面
2016/09/01 Javascript
利用node.js开发cli的完整步骤
2020/12/29 Javascript
python写入中英文字符串到文件的方法
2015/05/06 Python
Python实现基于二叉树存储结构的堆排序算法示例
2017/12/08 Python
Python简单计算文件MD5值的方法示例
2018/04/11 Python
python基于win32api实现键盘输入
2020/12/09 Python
html5嵌入内容_动力节点Java学院整理
2017/07/07 HTML / CSS
canvas线条的属性详解
2018/03/27 HTML / CSS
处理textarea中的换行和空格
2019/12/12 HTML / CSS
在线吉他课程,学习如何弹吉他:Fender Play
2019/02/28 全球购物
什么情况下你必须要把一个类定义为abstract的
2013/01/06 面试题
土木工程师岗位职责
2013/11/24 职场文书
村委会贫困证明
2014/01/14 职场文书
打造高效课堂实施方案
2014/03/22 职场文书
《新型玻璃》教学反思
2014/04/13 职场文书
项目建议书怎么写
2014/05/15 职场文书
党员领导干部承诺书
2014/05/28 职场文书
供用电专业求职信
2014/07/07 职场文书
经典毕业生求职信
2014/07/12 职场文书
国际贸易本科毕业生求职信
2014/09/26 职场文书
五一劳动节慰问信
2015/02/14 职场文书
2015年世界无烟日活动方案
2015/05/04 职场文书
防溺水主题班会教案
2015/08/12 职场文书
2016年“抗战胜利纪念日”71周年校园广播稿
2015/12/18 职场文书
一次线上mongo慢查询问题排查处理记录
2022/03/18 MongoDB
在Windows Server 2012上安装 .NET Framework 3.5 所遇到的问题
2022/04/29 Servers