JavaScript的坑(二)

2016-10-12  本文已影响30人  zyfEve

上一篇文章:JavaScript的坑(一)有兴趣的可以看看哦!

var a = 1;
function foo() {
    if (!a) {
        var a = 2;
    }
    alert(a);
};

foo(); // 输出2,不是1!

alert(a)在执行的时候,会去寻找变量a的位置,它从当前作用域开始向(或者说向外)一直查找到顶层作用域为止,若是找不到就报undefined。
因为在alert(a)的同级作用域里,我们再次声明了本地变量a,所以它报2;所以我们可以把本地变量a的声明向下(或者说向内)移动,这样alert(a)就找不到它了。像下面那样:

var a = 1;
function foo() {
    if (!a) {
        (function() {        // 这是 IIFE,它会创建一个新的函数作用域
            var a = 2;       // 并且该作用域在 foo() 的内部,所以 alert 访问不到
        }());               // 不过这个作用域可以访问上层作用域哦,这就叫:“闭包”
    };
    alert(a);
};
foo();//输出1

记住:JavaScript 只有函数作用域!

function test() {
    foo();
    //函数声明
    function foo() {
        alert("我是会出现的啦……");
    }
}
test();//输出我是会出现的啦……
function test() {
    foo();
    //函数表达式
    var foo = function() {
        alert("我不会出现的哦……");
    }
}
test();//报错:Uncaught TypeError: foo is not a function

在第一个例子里,函数 foo 是一个声明,既然是声明就会被提升。函数声明会连通命名和函数体一起被提升至作用域顶部。
在第二个例子里,被提升的仅仅是变量名foo,至于它的定义依然停留在原处。因此在执行foo()之前,作用域只知道foo的命名,不知道它到底是什么,所以执行会报错(通常会是:undefined is not a function)。这叫做函数表达式(Function Expression)

函数表达式只有命名会被提升,定义的函数体则不会。
函数声明会在js解析器解析时候率先解析,保证其他代码执行之前,函数可用。而函数表达式必须要等到代码解析器解析到他所在代码行,才会被解释执行。

上一篇下一篇

猜你喜欢

热点阅读