作用域和闭包的深入理解

作用域

高程92页

执行环境是 JavaScript中最为重要的一个概 念。

每个执行环境都有一个 与之关联的变量对象(variable object),环境中定义的所有变量和函数都保存在这个对象中。

每个函数都有自己的执行环境

当代码在一个函数中执行时,会创建变量对象的一个作用域链(scope chain)。

将函数的活动对象(activation object)作为变量对象。活动对象在最开始时只包含一个变量,即 arguments 对象(这个对象在全局环境中是不存在的)。

作用域链中 的下一个变量对象来自包含(外部)环境,而再下一个变量对象则来自下一个包含环境。这样,一直延 续到全局执行环境;全局执行环境的变量对象始终都是作用域链中的最后一个对象。

Javascript权威指南

函数的变量的作用域是在函数定义时确定的。

作用域在函数定义时确定很重要<br />这里一共有 执行环境 , 变量对象 , 函数 , 活动对象 几个专有名词

只提到函数情况 , 就不考虑全局环境了 , 一句话 :

每个函数有自己的执行环境 , 函数的活动对象作为执行环境的变量对象

作用域链中 的下一个变量对象来自包含(外部)环境,而再下一个变量对象则来自下一个包含环境。这样,一直延续到全局执行环境;全局执行环境的变量对象始终都是作用域链中的最后一个对象。

下面是举例

var color = "blue"; 
 
function changeColor(){     
    var anotherColor = "red"; 
    function swapColors(){    
        var tempColor = anotherColor;   
        anotherColor = color;  
        color = tempColor;
        debugger;
        // 这里可以访问 color、anotherColor 和 tempColor     
    } 
 
        // 这里可以访问 color 和 anotherColor,但不能访问 tempColor             
    swapColors(); 
} 
 
// 这里只能访问 color changeColor();

可以到浏览器控制台运行一下 .

我们看看debugger时 的 scoped

我们再加一层看看

var color = "blue"; 
 
function changeColor(){     
    var anotherColor = "red"; 
    function swapColors(){    
        var tempColor = anotherColor;   
        anotherColor = color;  
        color = tempColor;
        function consoleColors(){
            console.log(anotherColor);
            console.log(tempColor);
            debugger;
        }
        // 这里可以访问 color、anotherColor 和 tempColor
        consoleColors();
    } 
 
        // 这里可以访问 color 和 anotherColor,但不能访问 tempColor             
    swapColors(); 
} 
 
// 这里只能访问 color changeColor();
闭包变量
闭包变量

我们看到它保存了两个作用域链上父级和父级的父级(下面称为父父级)的变量(下文称闭包变量) , 并且是引用的变量才保存 , 也就是如果没有引用父级的变量 , 但是引用了父父级的变量 , 那么不会生成父级的变量对象而是生成父父级的变量对象 .

注意我上面说的是引用而不是保存 , 也就是说它并非是静态传递的 , 下面会提到 .

<a name="6be4fb54"></a>

闭包(Closure)

高程196页

闭包是指有权访问另一个函数作用域中的变量的函数。

另一个指哪一个 ?

创建闭包的常见方式,就是在一个函数内部创建另一个函数

**上面举的例子 , 就是在函数内部创建另一个函数 , 所以它生成了闭包 . 所以我们暂时把另一个当成上级就是了.

有权访问上级函数作用域中的变量的函数是闭包

我们甚至可以说一个函数 (在函数中定义&& 引用到了这个函数的变量 ) 它就是闭包 .

这是充分条件而非并必要条件 拓展会讲到特殊情况

Javascript权威指南

函数的变量的作用域是在函数定义时确定的。

理解闭包的关键:作用域是定义时上下文确定 , 而非依赖调用时的上下文<br />为什么会生成闭包变量(指作用域讲解中调试图的红框的Closure)? 因为 函数的作用域是定义时就确定了, 但是函数并不知道自己将会在哪里调用 , 它可能在父函数中调用 , 它可能在父函数外被调用 , 所以设计者就让它生成闭包变量来保存作用域链 , 保证它当在外部调用时 , 它就可以直接找到定义时状态的变量 . 当然无论是否是在外部调用的 , 他们都会生成闭包变量 , 他们都是闭包.

但是讨论不在父级作为返回值返回的闭包在父函数执行完毕后是没有意义的(刚刚在作用域链讲解示例就是这种情况) . 因为父级函数执行完毕之后 , 一般来说 , 父级的作用域链就被销毁了 , 外界访问不到闭包 , 自然也会被回收 .

但是返回值为函数时会有不同

在闭包从父级中被返回后,它的作用域链被初始化为包含父级函数的活动对象和全局变量对象

因为匿名函数的作用域链仍然在引用这个活动对象。换句话说,当父级函数返回后,其执行环境的作用域链会被销毁,但它的活动对象仍然会留在内存中;直到匿名函数被销毁后,父级的活动对象才会被销毁.

js引擎知道返回了一个闭包 , 它就不会销毁闭包中引用到的父级变量 . <br />所以闭包一般指下面的情况 将函数作为返回值给外界接收使用的情况

function createComparisonFunction(propertyName) {          
    return function(object1, object2){
            var value1 = object1[propertyName];
            var value2 = object2[propertyName];
            if (value1 < value2){
                return -1;
            } 
            else if (value1 > value2){
                return 1;
            } 
            else {
                return 0;
            }
        };
    }

它引用了父级的变量(参数也是变量) , 并作为父级的返回值返回给外层接收 .

闭包中变量的问题

刚刚说到闭包只是引用变量 , 而非按值传递

即闭包只能取得包含函数中任何变量的 后一个值。别忘了闭包所保存的是整个变量对象,而不是某个特殊的变量

比如以下函数

function createFunctions(){
    var result = new Array(); 
 
    for (var i=0; i < 10; i++){
        result[i] = function(){
            return i;
         };
    } 
    return result;
 }

它返回了一个闭包数组 , 这个函数原意认为 , 每个闭包都应该保存 i 应该是从0-9 ; 但是实际上他们都只保存了10 ; 因为他们引用的是同一个 i ;

<a name="e443a98d"></a>

拓展:深刻理解es6函数默认值作用域和其中生成中的闭包

阮一峰es6入门:函数的拓展--默认值的作用域

一旦设置了参数的默认值,函数进行声明初始化时,参数会形成一个单独的作用域(context)。等到初始化结束,这个作用域就会消失。这种语法行为,在不设置参数默认值时,是不会出现的。

Javascript权威指南

函数的变量的作用域是在函数定义时确定的。

**函数的变量的作用域是在函数定义时确定的 , 很重要 . **

var x = 1;
function foo(x, y = function() { x = 2; }) {
  var x = 3;
  y();
  console.log(x);
}

这个函数会打印出什么 ? 按照常识我们会认为它打印出了 2 ;按常识 , 在这个上下文中 , 作用域链会往上找 , x 应该指向的是var x = 3 的 x ; 被 y() 后改为了 2 ;

但是它打印出了 3 而非 2;

奥秘就在开头 , 如果设置默认值时 , 就会生成一个单独的作用域 就好像是这样子

function(){
    let x;
    let y = function(){
        x = 2;
    }
}

它类似于一个函数 (下面称作用域伪函数), 生成了一个作用域 , 它与foo函数是平行的 , 而非包含关系

所以 y 是 这个作用域中的闭包 , 我们可以打上debugger来看看

var x = 1;
function foo(x, y = function() { x = 2;debugger; }) {
  var x = 3;
  y();
  console.log(x);
}
闭包变量
闭包变量

注意x = 2 ; y()把父级作用域中的 x 由undefined 变为了 2 ;

同时y也非常特殊 , 它并非是被 return 而保存下来的 , 它是编译器内部把y的地址赋给了参数(如果你没有输入)

为什么他们是平行的 以下两个例子说明

//仍然是上面的函数
y(5);
//x已经不取默认值

//5
//Closure(foo)
    x:2

我们传入x的参数 , 很显然编辑器并不考虑有没有传参数 , 传了几个参数 , 只要你设置了其中一个默认值 它就会在执行函数前一步执行作用域伪函数: 声明所有参数变量 有默认值则赋值 无默认值则是undefined ;

如果你传入了某个参数 , 它就抛弃这个值 ; 如果你没有传入某个参数 , 它就把这个值赋给这个参数 ;

函数是引用变量 , 所以这也是没有return却可以使用一个闭包的原因

下面是第二个例子

var x = 1;
function foo(y = function() { x = 2;}) {
  var x = 3;
  y();
  console.log(x);
}
console.log(x);

这次我们没有定义 x 了 .

会发生什么?

A. y()时寻找到foo()内部的x变量并赋值

B.y()时寻找到外部的x变量并赋值

C.定义时报错

D.y()执行时报错

答案是

B
解析:虽然在某个内部函数被调用但是它还是沿定义时作用域链向上找变量 , 所以 没有x参数 , 它就找到了全局变量的x .

//3
//2

这个例子就是与foo函数是作用域平行的最好证明 .

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,992评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,212评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,535评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,197评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,310评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,383评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,409评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,191评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,621评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,910评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,084评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,763评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,403评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,083评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,318评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,946评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,967评论 2 351

推荐阅读更多精彩内容