下面小编就为大家带来一篇浅谈 JavaScript: 执行环境、作用域及垃圾回收。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
Javascript 是一种由 Netscape 的 LiveScript 发展而来的原型化继承的基于对象的动态类型的区分大小写的客户端脚本语言,主要目的是为了解决服务器端语言,比如 Perl,遗留的速度问题,为客户提供更流畅的浏览效果。
执行环境定义了变量或函数有权访问的其他数据,决定了它们各自的行为。每个执行环境都有一个与之关联的变量对象。
全局执行环境是最外围的一个执行环境。根据 JavaScript 实现所在的宿主环境不同,表示执行环境的对象也不一样。在 web 浏览器中,全局执行环境被认为是 window 对象。因此,所有的全局变量和函数都是作为 window 对象的属性和方法创建的。
变量对象:环境中定义的所有变量和函数都保存在这个对象中。
作用域链:当代码在一个环境中执行时,会创建变量对象的一个作用域链。作用域链的用途是保证对执行环境有权访问的所有变量和函数的有序访问。作用域链的前端,始终都是当前执行的代码所在环境的变量对象。
活动对象:活动对象在最开始时只包含一个变量,即 arguments 对象。作用域链中的下一个变量对象来自包含 (外部) 环境,而再下一个变量对象来自下一个包含环境。这样一直延续到全局执行环境;全局执行环境的变量对象始终都是作用域链中的最后一个对象。
标识符解析:标识符解析是沿着作用域链一级一级地搜索标识符的过程。搜索过程始终从作用域链的前端开始,然后逐级地向后回溯,直至找到标识符为止。
示例代码:
- var color = "blue";
- function changeColor() {
- if (color === "blue") {
- color = "red";
- } else {
- color = "blue";
- }
- }
- changeColor();
- alert("Color is now " + color);
函数 changeColor()的作用域链包含两个对象:它自己的变量对象 (其中定义着 arguments 对象) 和全局变量的变量对象。可以在函数内部访问变量 color,就是因为可以在这个作用域链中找到它。
此外,在局部作用域中定义的变量可以在局部环境中与全局变量互换使用,示例:
- var color = "blue";
- function changeColor() {
- var anotherColor = "red";
- function swapColors() {
- var tempColor = anotherColor;
- anotherColor = color;
- color = tempColor;
- // 这里可以访问color、anotherColor和tempColor
- }
- // 这里可以访问color、anotherColor,不能访问tempColor
- swapColors();
- }
- // 这里只能访问color
- changeColor();
以上代码供涉及 3 个执行环境:全局环境、changeColor() 的句柄环境和 swapColors() 的局部环境。
全局变量中有一个变量 color 和一个函数 changeColor()。changeColor() 的局部变量中包含了一个变量 anotherColor 和一个函数 swapColors() 函数,它可以访问全局变量中的 color。swapColors() 的局部变量中有一个变量 tempColor。在 swapColors() 中可以访问全局变量中的 color,也可以访问 anotherColor 变量,因为那两个环境是它的父执行环境。上面的例子的作用域链为:
其中,内部环境可以通过作用域链访问所有的外部环境,但外部环境不能访问内部环境中的任何变量和函数。环境变量之间的联系是线性的、有次序的。每个变量只能向上级搜索作用域链,以查询变量和函数名,即首先在本作用于中查询变量或函数名,如果没有再向上一级作用域链查询,直到顶级作用域。但是任何环境都不能向下搜索作用域链而进入另一个执行环境。
函数参数也被当作变量来对待,因此其访问规则与执行环境中的其他变量相同。
1. 延长作用域链
当执行流进入下列任何一个语句时,作用域链就会得到延长:
• try-catch 语句的 catch 块
• with 语句
这两个语句会在作用域的前端添加一个变量对象。
对于 with 语句来说,会将指定的变量添加到作用域链中。对 catch 语句来说,会创建一个新的变量对象,其中包含的是被抛出的错误对象的声明。
举个例子:
- function buildUrl() {
- var qs = "?debug=true";
- with(location) {
- var url = href + qs;
- }
- return url;
- }
with 语句接收的是 location 对象,因此其变量对象中包含了 location 对象的所用属性和方法,这个变量对象被添加到作用域链的前端。当在 with 语句中引用变量 href 时(实际引用的是 location.href), 可以在当前环境变量中找到。当引用变量 qs 时,引用的是 buildUrl() 中定义的那个变量,该变量位于函数环境变量对象中。至于 with 语句内部,则定义了一个名为 url 的变量,因而 url 就成了函数执行环境的一部分,可以作为函数的值被返回。
2. 没有块级作用域
在 JavaScript 中,封闭的花括号没有自己的作用域。看下面的代码:
- if (true) {
- var color = "blue";
- }
- alert(color); // "blue"
在 JavaScript 中,if/for 语句创建的变量声明会将变量添加到当前的执行环境中。例如:
- for (var i = 0; i < 10; i++) {
- doSomething(i);
- }
- alert(i); // 10
垃圾回收
与 Java 相似,JavaScript 也具有自动回收垃圾机制。执行环境会负责管理代码执行过程中使用的内存。在编写程序时,不需要关系内存使用问题,所需内存的分配以及无用内存的回收完全实现了自动管理。垃圾回收机制的原理就是:找出不再继续使用的变量,然后释放其占用的内存。为此,垃圾回收器会按照固定的时间间隔 (或代码执行中预定的收集时间),周期性地进行这一操作。
在做垃圾回收之前,必须判断该资源是否无用,对于不再使用的变量打上标记,以备将来回收其内存。用于标识无用变量的策略通常有两个实现。
1 标记清除
JavaScript 中最常用的垃圾收集方式是标记清除。当变量进入环境,就将变量标记为 "进入环境";当变量离开环境时,则将变量标记为 "离开环境"。垃圾回收器在运行的时候会给所用变量都加上标记。然后,它会去掉环境中的变量以及被环境中的变量引用的变量的标记。而在此之后再被加上标记的变量将被视为准备删除的变量,最后垃圾回收器完成内存清除工作,销毁带标记的值并回收它们所占的内存空间。
2. 引用计数
引用计数是指跟踪记录每个值被引用的次数。当声明了一个变量并将一个引用类型值赋给该变量时,则这个值的引用次数就是 1。如果同一个值又被赋给另一个变量,则该值的引用次数加 1。相反,如果包含这个值引用的变量又取得了另一个变量,则这个值的引用次数减 1。当这个变量的引用次数为 0 时,则说明没有办法再引用这个变量了,因而就可以将其内存空间回收回来。当垃圾回收器下次运行时就会回收这些引用次数为零的值占用的内存。
引用计数会产生的一个问题就是可能会导致循环引用。例如:
- function problem() {
- var objA = new Object();
- var objB = new Object();
- objA.someOtherObj = objB;
- objB.someOtherObj = objA;
- }
上面的例子中,objA 和 objB 通过属性相互引用。函数执行完成后,objA 和 objB 将继续存在,它们的引用计数不会为 0。这种情况会导致 objA 和 objB 所占的内存无法回收。
以上这篇浅谈 JavaScript: 执行环境、作用域及垃圾回收就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持 phperz。
来源: http://www.phperz.com/article/17/0331/265269.html