举一个简单的闭包例子:
function A() {
var i = 1;
var j = 2;
return function(){
return i;
}();
}
var B = A();
在语句 return i
这一行设置断点,调试如下:
之前看过有关闭包的资料时知道由于函数 A
中的变量被引用,所以函数 A
不会被 GC
回收,那么这个不回收指的是整个函数 A
都不会被回收,还是被直接引用的部分不会被回收呢?
再举一例:
function A() {
var i = {x : 1};
var j = i;
return function(){
return j;
}();
}
var B = A();
此时,i
被 j
引用,所以 i
和 j
指向同一个内存空间,但是断点调试时依然访问不到未被直接引用的变量 i
:
我的猜想是,当我设置断点进行调试时由于是全局作用域,所以我访问不到外层函数内部未被闭包暴露在全局作用域下的变量,而并不是因为该变量已经被 GC
回收了。
那么我最终的问题是:未被闭包直接引用的外层函数的变量在我设断点调试时到底有没有被 GC
回收呢?如果未被回收,为何我调试时访问不到呢?是因为作用域的问题吗?如果此时未被回收,那么该变量何时会被回收呢?
Firefox
和 Safari
调试时居然是可以访问到未引用的变量 j
的,截图如下:
1.Firefox
:
2.Safari
:
现在基本明白了,可能是 debugger 的问题?
测试代码如下:
var v0 = 'i am at level 0';
var f1 = function () {
var v1 = 'i am at level 1';
var f2 = function (){
var i = 1;
var j = 2;
var f3 = function (){
console.log(i);
}
f3();
}
f2();
}
f1();
注意右侧的方框,你为什么访问不到
j
?因为你的断点在
function(){...}();
的作用域里,而不是你认为的全局环境所以你可以访问到自身的
变量abc
(我为了举例加上的),闭包中的a
(你问题中的变量i
)访问不着
j
因为你没使用它,Closure里就没有它啊至于GC机制是不是立刻回收的,我不了解就不说了..