JS三座大山再学习 ---- 做用域和闭包

本文已发布在西瓜君的我的博客,原文传送门javascript

做用域

JS中有两种做用域:全局做用域|局部做用域html

栗子1

console.log(name);      //undefined
var name = '波妞';
var like = '宗介'
console.log(name);      //波妞
function fun(){
    console.log(name);  //波妞
    console.log(eat)    //ReferenceError: eat is not defined
    (function(){
        console.log(like)   //宗介
        var eat = '肉'
    })()
}
fun();
  1. name定义在全局,在全局能够访问到,因此 (2) 打印可以正确打印;
  2. 在函数fun中,若是没有定义name属性,那么会到它的父做用域去找,因此 (3) 也能正确打印。
  3. 内部环境能够经过做用域链访问全部外部环境,但外部环境不能访问内部环境的任何变量和函数。相似单向透明,这就是做用域链,因此 (4) 不行而 (5) 能够。

那么问题来了,为何第一个打印是"undefined",而不是"ReferenceError: name is not defined"。原理简单的说就是JS的变量提高java

变量提高:JS在解析代码时,会将全部的声明提早到所在做用域的最前面数组


栗子2

console.log(name);      //undefined
var name = '波妞';
console.log(name);      //波妞
function fun(){
    console.log(name)   //undefined
    console.log(like)   //undefined
    var name = '大西瓜';
    var like = '宗介'
}
fun();

至关于markdown

var name;
console.log(name);      //undefined
name = '波妞';
console.log(name);      //波妞
function fun(){
    var name;
    var like;
    console.log(name)   //undefined
    console.log(like)   //undefined
    name = '大西瓜';
    like = '宗介'
    console.log(name)   //大西瓜
    console.log(like)   //宗介
}
fun();

注意:是提早到当前做用域的最前面闭包


栗子3

printName();     //printName is not a function
var printName = function(){
    console.log('波妞')
}
printName();       //波妞

至关于函数

var printName;
printName();     //printName is not a function
printName = function(){
    console.log('波妞')
}
printName();       //波妞

这样一来就好理解了,函数表达式在声明的时候还只是个变量post


栗子4

{
    var name = '波妞';
}
console.log(name)   //波妞

(function(){
    var name = '波妞';
})()
console.log(name)   //ReferenceError: name is not defined

{
    let name = '波妞';
}
console.log(name)   //ReferenceError: name is not defined

从上面的栗子能够看出,不能够草率的认为JS中var声明的变量的做用范围就是大括号的起止范围,ES5并无块级做用域,实质是函数做用域;ES6中有了let、const定义后,才有了块级做用域。ui


栗子5

function p1() { 
    console.log(1);
}
function p2() { 
    console.log(2);
}
(function () { 
    if (false) {
        function p1() {
            console.log(3);
        }
    }else{
        function p2(){
            console.log(4)
        }
    }
    p2();
    p1()
})();       
//4
//TypeError: print is not a function

这是一个很是经典的栗子,声明提早了,可是由于判断条件为否,因此没有执行函数体。因此会出现"TypeError: print is not a function"。while,switch,for同理spa

闭包

函数与对其状态即词法环境(lexical environment)的引用共同构成闭包(closure)。也就是说,闭包可让你从内部函数访问外部函数做用域。在JavaScript中,函数在每次建立时生成闭包。

上面的定义来自MDN,简单讲,闭包就是指有权访问另外一个函数做用域中变量的函数。


  • 闭包的关键在于:外部函数调用以后其变量对象本应该被销毁,但闭包的存在使咱们仍然能够访问外部函数的变量对象.,
//举个例子
function makeFunc() {
    var name = "波妞";
    function displayName() {
        console.log(name);
    }
    return displayName;
}

var myFunc = makeFunc();
myFunc();

JavaScript中的函数会造成闭包。 闭包是由函数以及建立该函数的词法环境组合而成。这个环境包含了这个闭包建立时所能访问的全部局部变量

在例子中,myFunc 是执行 makeFunc 时建立的 displayName 函数实例的引用,而 displayName 实例仍可访问其词法做用域中的变量,便可以访问到 name 。由此,当 myFunc 被调用时,name 仍可被访问,其值 '波妞' 就被传递到console.log中。建立闭包最多见方式,就是在一个函数内部建立另外一个函数


  • 一般,函数的做用域及其全部变量都会在函数执行结束后被销毁。可是,在建立了一个闭包之后,这个函数的做用域就会一直保存到闭包不存在为止
//例二
function makeAdder(x) {
  return function(y) {
    return x + y;
  };
}

var add5 = makeAdder(5);
var add10 = makeAdder(10);

console.log(add5(2));  // 7
console.log(add10(2)); // 12

//释放对闭包的引用
add5 = null;
add10 = null;

从本质上讲,makeAdder 是一个函数工厂 — 他建立了将指定的值和它的参数相加求和的函数。在上面的示例中,咱们使用函数工厂建立了两个新函数 — 一个将其参数和 5 求和,另外一个和 10 求和。

add5 和 add10 都是闭包。它们共享相同的函数定义,可是保存了不一样的词法环境。在 add5 的环境中,x 为 5。而在 add10 中,x 则为 10。

闭包的做用域链包含着它本身的做用域,以及包含它的函数的做用域和全局做用域。


  • 闭包只能取得包含函数中的任何变量的最后一个值
//栗子1
function arrFun1(){
    var arr = [];
    for(var i = 0 ; i < 10 ; i++){
        arr[i] = function(){
            return i
        }
    }
    return arr
}
console.log(arrFun1()[9]());     //10
console.log(arrFun1()[1]());     //10

//栗子2
function arrFun2(){
    var arr = [];
    for(var i = 0 ; i < 10 ; i++){
        arr[i] = function(num){
            return function(){
                return num
            };
        }(i)
    }
    return arr
}
console.log(arrFun2()[9]());     //9
console.log(arrFun2()[1]());     //1

栗子 1 中,arr数组中包含10个匿名函数,每一个函数均可以访问外部的变量 i , arrFun1 执行后,其做用域被销毁,但它的变量依然存在内存中,能被循环中的匿名函数访问,这是的 i 为 10;

栗子 2 中,arr数组中有是个匿名函数,其匿名函数内还有匿名函数,最内层匿名函数访问的 num 被 上一级匿名函数保存在了内存中,因此能够访问到每次的 i 的值。


若有错误,请斧正

以上

 

 

出处:http://www.javashuo.com/article/p-gflavopm-dq.html

相关文章
相关标签/搜索