zoukankan      html  css  js  c++  java
  • JavaScript的作用域和闭包

    对于初学者而言,可以这样理解: JS只有两种作用域

    1. 全局作用域

    2. 函数作用域 

    必须牢记一点 JS中没有块级作用域

    {
        var test=10;
    }
    console.log(test);    // 控制台输出:10

    再看一个

    var obj={
        test:10,
        myFunc:function(){
            console.log(test); 
        }
    };
    obj.myfunc();   // 出错,或者IDE直接报警了

    这个就更难理解了吧

     写在函数体内的变量,只要不是嵌套在更深层的函数里,就是处在同一作用域的,“互相可见”;而其他的花括号,不管是for后跟的,if后跟的,还是对象字面量的,一概“不作数”,起不到定义作用域的效果,变量声明写在那些花括号的里面或外面都一样。

    那位于嵌套的函数里的作用域呢?它们享有“单向透明”的特权,即:在较内层次的作用域内可以访问较外层次作用域里的变量,反之则不行。

    function outerFunc(){
        for(var i=0;i<10;i++){doSomething;}
        console.log(i); // 控制台输出10,因为i位于outerFunc的作用域
        var outer = 10;
        function innerFunc(){
            var inner = outer; // 内层作用域可以访问外层作用域里的变量
        }
        console.log(inner); // 报错,外层作用域访问不到内层作用域里的变量
    }

    再来分析上一个例子。我们试图在myFunc的作用域内部访问test,然而test并不是一个“与myFunc位于同一个对象作用域”的变量,事实上根本不存在“对象作用域”这回事,test是obj的一个属性,不是一个“独立”的变量,要访问test只能通过点运算符obj.test或obj["test"],哪怕是在myFunc内部。当然,myFunc内部可以访问到obj这个位于外层作用域的变量,没有问题。于是将代码改写如下:

    var obj={
        test:10,
        myFunc:function(){
            console.log(obj.test); 
        }
    };
    obj.myfunc();   // 10

    既然在内层作用域里可以访问外层作用域,那么就产生了一个有趣的现象,叫做“闭包”。制造一个闭包只需要两步:

    1.在内层函数里引用外层函数的变量

    2.将内层函数作为外层函数的返回值返回出去

    function outer(){
        var test = 10;
        var inner = function(){
            console.log(test++);
        };
        return inner;
    }
    
    var myFunc = outer(); // 将outer的返回值(inner函数)赋给myFunc
    myFunc(); // 10
    myFunc(); // 11
    myFunc(); // 12

    这个被返回的inner函数就是一个闭包。虽然outer函数运行结束了,但它的内部变量test因为被闭包引用,所以并没有被销毁,而是被保存了起来,并且可以通过闭包继续操作。当然,外界永远无法访问test这个变量,它成了inner(以及myFunc)所指向的函数的“私有变量”。

    本文转载自 http://www.cnblogs.com/leegent/p/5377177.html

    在ES6中新增加了块级作用域( {.....} ), 通过使用let关键字,能够让你声明的变量属于一个单独的代码块:

    function foo() {
        
             var a = 1;
    
              if(a >= 1) {
    
                let b = 2;
    
                while (b < 5) {
    
                  let c = b * 2;
                  b ++;
    
                  console.log(a + c); 
    
     
                 }
    
    
               }
    }

    foo();

    由于使用let代替了var, b只属于if语句,而不是整个foo()的函数作用域. 同样, c也只属于while循环. 

  • 相关阅读:
    4.2.1 B
    4.1.1 A
    C
    A
    排序(sort qsort)
    晕,
    clipssubviews = clipstobounds
    scrollview once more,滑出来的 刚好等于 上下偏移的,
    关于 层的显示,
    水倒过来,倒过去,穷折腾啊,
  • 原文地址:https://www.cnblogs.com/wphl-27/p/8491327.html
Copyright © 2011-2022 走看看