2018-01-07 关于javascript闭包和作用域的理解

关于 javascript 闭包的一些思考

什么是作用域?

  • 作用域
    • 众所周知 在javascript 作用域就是限制我们执行代码的一个范围,或者说是框架。
    • 首先来谈谈js的编译原理,其中不可避免的就要提到 引擎、编译器、作用域
      • 引擎:负责整个 JavaScript 程序的编译及执行过程
      • 编译器:负责语法分析及代码生成
      • 作用域: 负责收集并维护由所有声明的标识符(变量)组成的一系列查 询,并实施一套非常严格的规则,确定当前执行的代码对这些标识符的访问权限
    • 这次我们的主角是==作用域==
    • 下面我们来看一个简单的赋值操作:

      var a = 2;
      
      • 然后他们三个都干了什么啦?
        • 变量的赋值操作会执行两个动作,首先编译器会在当前作用域中声明一个变量(如 果之前没有声明过),然后在运行时引擎会在作用域中查找该变量,如果能够找到就会对 它赋值。
    • 当然我们不必深究他们到底干了什么,我们要深入探究 作用域到底是什么?

什么是词法作用域?

  • 词法作用域
    • 我们来先看看一段代码

      function foo(a) { 
          var b = a * 2;
          function bar(c) { 
              console.log( a, b, c );
          }
          bar( b * 3 ); 
          
      }
      foo( 2 ); // 2, 4, 12
      
      • 1 包含着整个全局作用域,其中只有一个标识符:foo。

      • 2 包含着 foo 所创建的作用域,其中有三个标识符:a、bar 和 b。

      • 3 包含着 bar 所创建的作用域,其中只有一个标识符:c。

    • 上面描述的就是作用域的作用,每个标识符都对应着相应的。我们==把作用域看做一个气泡==

什么是函数作用域?

  • 函数作用域
    • 如同上面的词法作用域那样,在 javascript 中当我们创建一个函数的时候都会创建一个新的作用域。
    function foo(a){
        var b = 2;
        
        //一些代码
        function bar(){
            // ...
        }
        
        // 更多的代码
        
        var a = 3;
    }
    
    
    • 在这个代码片段中,==foo(..) 的作用域气泡中包含了标识符 a、b、c 和 bar==。无论标识符 声明出现在作用域中的何处,这个标识符所代表的变量或函数都将附属于所处作用域的气泡。

什么是块作用域?

  • 块作用域
    • 除 JavaScript 外的很多编程语言都支持块作用域,因此其他语言的开发者对于相关的思维 方式会很熟悉,但是对于主要使用 JavaScript 的开发者来说,这个概念会很陌生。
      • 看下面的代码
      for (var i=0; i<10; i++) { 
          console.log(i)
      }
      
      • 我们在for的头部定义了变量 i ,而且我们只想在for循环中使用 i ,而忽略了在 javascript 中 i 会绑定在全局变量(外部作用域)中。
      • 在 ES6 中我们推荐使用 let 来避免 i收到全局变量的污染

        for(let i = 0;i < 10; i++ ){
            console.log(i);
        }
        

什么是垃圾回收机制

  • 垃圾回收机制
    • JavaScript 垃圾回收的机制很简单:找出==不再使用的变量==,然后释放掉其占用的内存,但是这个过程不是时时的,因为其开销比较大,所以垃圾回收器会按照固定的时间间隔周期性的执行。
    • 那什么是==不再使用的变量==啦?
      • 我们知道js中的全局变量,和局部变量。全局变量在浏览器页面卸载的时候才会回收。而局部变量在函数生命周期结束的时候浏览器为了节约内存空间,就需要回收这一变量。
    • 一种回收方法-标记清除(mark and sweep)
      • 这是JavaScript最常见的垃圾回收方式,当变量进入执行环境的时候,比如函数中声明一个变量,垃圾回收器将其标记为“进入环境”,当变量离开环境的时候(函数执行结束)将其标记为“离开环境”。
    • 还有其他的回收的方法就不多多探究了。

什么是闭包?

  • 闭包的理解
    • 作用域闭包

      • 参考阮一峰大神的文章,代码中的f2函数,就是闭包。
      function f1(){
         var n = 999;
         function f2(){
             alert(n);
         }
         return f2;
      }
      
      var result = f1();
      result();//999
      
      • 闭包就是能够读取其他函数内部变量的函数。

      • 由于在Javascript语言中,只有函数内部的子函数才能读取局部变量,因此可以把闭包简单理解成"定义在一个函数内部的函数"。

      • 所以,在本质上,闭包就是将函数内部和函数外部连接起来的一座桥梁。

    • 闭包的用途

      • 一个是前面提到的可以读取函数内部的变量
      • 另一个就是让这些变量的值始终保持在内存中。
      function f1(){
      
          var n=999;
      
          nAdd=function(){n+=1}
      
          function f2(){
                alert(n);
          }
      
          return f2;  
          
      }
      
      var result=f1();
      
      result(); // 999
      
      nAdd();
      
      result(); // 1000
      
      • result实际上就是闭包f2函数。它一共运行了两次,第一次的值是999,第二次的值是1000。这证明了,函数f1中的局部变量n一直保存在内存中,并没有在f1调用后被自动清除。
      • 原因就在于f1是f2的父函数,而f2被赋给了一个全局变量,这导致f2始终在内存中,而f2的存在依赖于f1,因此f1也始终在内存中,不会在调用结束后,被垃圾回收机制(garbage collection)回收。

参考:

你不知道的javascript(上)

学习Javascript闭包(Closure)

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

推荐阅读更多精彩内容