【跟着犀牛书复习JS基础】作用域、作用域链、闭包和执行上下文(this)

引言

可以说是取名废了,把这几个关键词放在一起也是因为看完犀牛书相关章节和很多技术文章之后觉得这几个概念是相互渗透的,需要放在一起理解。而在这之前,我对这几个概念都是久闻其名,真正遇到了似懂非懂还很心虚,每每看完一篇零零散散的技术文章总觉得明白了,但是下一次遇到了还是不明白。这种其实充其量只能说记住了一些知识点碎片,而学习不能只是碎片,必须是找到碎片与碎片之间的关联与规律,才能将知识碎片连成完整的知识体系吃下去。所以,某位大佬说得对,有产出的学习才是真的学习,毕竟如果不是真的搞清楚了,你什么也写不出来,何况写一遍还能加深印象。如果你也有类似的情况,请重视起来,一起加油叭!

本章对标犀牛书第6版3.9变量声明、3.10变量作用域和第8大章8.6之前的章节,之后两小节值得在后面的文章里更详细的单独研究!

变量声明和函数定义

你可能觉得这过于基础了,但事实上如果忽略其中的一些细节会影响后面的理解。

变量声明

我们知道在ES6之前,我们通常用var关键字来声明一个或多个变量var i,sum;,还可以将变量的初始赋值和变量声明写在一起var i = 0, j = 1, k = 2;,如果没有在var声明语句中给变量指定初始值,那么虽然声明了变量,但在给他赋值之前,它的值就是undefined。

如果试图读取一个没有声明的变量,会触发一个报错。在严格模式下,给一个未声明的变量赋值也会触发一个报错,在非严格模式下,给一个未声明的变量赋值,js实际上会给全局对象创建一个同名属性,使其工作起来一个正常声明的全局变量。

var声明的属性是不可配置的,也就是不可以用delete操作符删除,而给未声明的变量赋值创建的全局对象同名属性是正常的可配置属性,是可以被删除的

函数定义

函数定义可以通过两种方式来进行:

  • 函数定义表达式 var func = function(){...};
  • 函数声明function func(){...}

虽然两者都包含相同的函数名,都创建了一个新的函数对象,但还是有所区别,区别就体现在声明提升的时候。

作用域和声明提升

全局作用域和函数作用域

ES6之前,JS是没有常用的不受约束的块级作用域的,只有全局作用域和函数作用域。全局作用域是顶级作用域,顾名思义,在顶级声明的所有变量全局中都可访问;函数作用域是指在函数内声明的所有变量在函数体内始终可见,包括嵌套函数内。以及,函数体内的局部变量优先级高于同名的全局变量,如果函数内声明的全局变量或函数参数带有的变量和全局变量同名,那么全局变量就被局部变量覆盖。例如:

var scope = 'global';
function f() {
  var scope = 'local';
  console.log(scope); //local 同名的全局变量被局部变量覆盖
  function f2() {
    console.log(scope); //local 函数体内声明的变量在嵌套函数内也可见
  }
  f2();
}
f(); 

声明提升

正常情况下我们认为js语句是由上到下一句一句执行的,这完全没错,但是有一种情况会使我们很迷惑。

a = 2;
var a;
console.log(a); //2

console.log(a); //undefined
var a = 2; 

实际上这是因为包括变量和函数在内的所有声明(但不涉及赋值)都会在如何代码被执行前也就是js的预编译阶段被首先处理,会被”提升“到相应作用域的顶部,这个过程就是声明提升,关于声明提升,我们只需要注意以下几点:

  1. 函数声明也会被提升,但函数表达式不会。也就是说上述两种函数定义的方式,使用函数声明语句,函数变量名和函数体均会提升,但使用var的表达式,只是变量声明被提升,函数体会留在原来的位置等待执行;

    f();
    
    function f(){
      console.log(a);  //会被执行且输出undefined
      var a = 2;
    }
    
    f(); //TypeError f is not a function
    var f = function bar() {
      console.log(a);  
      var a = 2;
    }
    
  2. var定义的具名的函数表达式,函数名变量也不会被提升;

    f(); //TypeError f is not a function
    bar(); //ReferenceError: bar is not defined
    var f = function bar() {
      console.log(a);  
      var a = 2;
    }
    //注意这里f()抛出的是TypeError而不是ReferenceError 是因为变量名已经被提升但没有被赋值,所以此时f是undefined,对undefined做调用执行,因此抛TypeError异常。
    //这段代码实际上被编译成以下形式:
    var f;
    f(); //TypeError
    bar(); //ReferenceError
    f = function() {
      var bar = self;
      var a
      a = 2;
    }
    
  3. 函数声明优先,已知变量声明和函数声明都会被提升,如果同一作用域里有相同命名的函数声明和变量声明提升,函数声明优先。

    foo();  // foo2
    var foo = function() {
        console.log('foo1');
    }
    
    foo();  // foo1,foo重新赋值
    
    function foo() {
        console.log('foo2');
    }
    
    foo(); // foo1
    
    //这段代码实际上被形容成
    function foo() {
      console.log('foo2');
    }
    foo();
    foo = function() {
      console.log('foo1');
    }
    foo();
    foo();
    //尽管var foo出现在function foo(){..}之前,但是函数声明会被提升到普通变量声明之前,因此重复的声明会被忽略,但会被重新赋值。此外,后面的函数声明还是可以覆盖前面的,比如:
    foo(); // foo2
    function foo() {
      console.log('foo1');
    }
    function foo() {
      console.log('foo2');
    }
    
  4. 条件语句中的函数声明:

    这个东西迷惑了我好久!!!浪费我的时间!!!

    MDN中的相关解释:https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Statements/function

    Segmentfault中回复楼的相关解释:https://segmentfault.com/q/1010000005337550

    以及下面的(【阮一峰】ES6入门)中也有相关解释:【阮一峰】ES6入门

    foo();
    var a = true;
    if(a) {
     function foo(){console.log('foo1')}
    } else {
     function foo(){console.log('foo2')}
    }
    //理论上,这段代码会输出foo2
    //经实验,实际上在大多数浏览器控制台里会抛出TypeError foo is not a function,在Safari里输出了foo2
    
    (function () {
      if (false) {
        // 重复声明一次函数f
        function f() { console.log('I am inside!'); }
      }
      f(); //TypeError foo is not a function
    }());
    
    //总之,在块级作用域和条件语句中声明函数,在不同浏览器中解释的标准都不一样,考虑到环境导致的行为差异太大,不要在块级作用域或条件语句中声明函数!!!
    // 等我有时间了再来纠结这个吧。。。。。。不过感觉遥遥无期。。。。。
    
    

块级作用域

偷个懒~ 【阮一峰】ES6入门

注释一点:

绝大部分的文章里都提到在ES6之前,js中是没有块级作用域的,只有函数作用域和全局作用域,实际上,ES3中的try/catch语句中的catch语句就会创建一个块级作用域

try {
  throw 2
} catch (a) {
  console.log(a); //2
}
console.log(a);//Reference Error

我试着在babel在线编译器里转换

{
  let a = 2;
  console.log(a);
}
console.log(a);

但是得到的是

"use strict";

{
var _a = 2;
console.log(_a);
}
console.log(a);

作用域链和闭包

作用域链

说作用域链先理解一下自由变量:当前作用域没有定义的变量即自由变量。如何访问到自由变量:向父级作用域寻找。如果父级作用域没有,则一层一层向上寻找,直到找到全局作用域,如果还是没找到,则宣布放弃。

这个一层一层向上的层级关系,就是作用域链,它是一个对象列表。

只用记住一点:自由变量将从作用域链中去寻找, 依据的是函数定义时的作用域链,而不是函数执行时,JS采用的是静态作用域,依据的是函数定义时的位置。

function F1() {
    var a = 100
    return function () {
        console.log(a)
    }
}
var f1 = F1()
var a = 200
f1() //100

闭包

当函数可以记住并访问函数体内的变量时,就形成了闭包,因此严格来说每个函数都是闭包。

注意三点:

  1. 同一个作用域链中的多个闭包共享私有变量或变量

    function countFunc() {
     var funcs = [];
     for(var i = 0; i < 10; i++){
        funcs[i] = function (){ return i }
      }
      return funcs
    }
    countFunc()[0]() //10
    //这段代码创建了10个闭包,但是都是在一个函数调用中定义的,因此它们共享变量i
    
  2. 每次调用js函数的时候,都会创建一个新的作用域链

    function counter() {
      var n = 0;
      return {
        counter: function() { return n++; },
        reset: function() { n = 0; }
      }
    }
    var c = counter();
    var d = counter();
    c.counter(); //0
    d.counter(); //0 互不干扰
    c.reset();  //重置c
    c.counter(); //c的reset和counter共享n变量
    
  3. this是JS关键字而不是变量,每一个函数调用都包含一个this值,如果闭包在外部函数中,是无法访问外部函数的this值的,arguments同理,虽然它不是关键字,但是调用函数时会自动声明它。

执行上下文this

记住大佬做的一张图就够了(侵删):

img

小结


有点曲折的一章,被条件语句和块级作用域中的函数声明搞得头大。

犀牛书没有把这一块说的很清楚,看了《你不知道的JS》相关章节作补充

后续需要补充:js编译执行机制,this详解,面试题详解

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

推荐阅读更多精彩内容