我们先来看一个题目:
第一个script里可以看出var a 被提升到顶部,a = 'littlebear'被保留在原地。
第二个script,之所以不先打印undefined ,是因为a在上面已经被var声明过,所以var a不会再次被提升。
再看一个题目:
可以看到function a(){}被提升到最顶端。说明函数的提升变量的提升
1.变量提升
在ES6之前,JavaScript没有块级作用域(一对花括号{}即为一个块级作用域),只有全局作 用域和函数作用域。变量提升即将变量声明提升到它所在作用域的最开始的部分。 上个简历的例子如:
console.log(a); // undefined
var a = 'aaa';
console.log(a); // aaa
}
fn();
之所以会是以上的打印结果,是由于js的变量提升,实际上上面的代码是按照以下来执行的:
2.函数提升
js中创建函数有两种方式:函数声明式和函数字面量式。只有函数声明才存在函数提升!如:
之所以会有以上的打印结果,是由于js中的函数提升导致代码实际上是按照以下来执行的:
3.什么是闭包
简单的说,Javascript允许使用内部函数---即函数定义和函数表达式位于另一个函数的函数体内。而且,这些内部函数可以访问它们所在的外部函数中声明的所有局部变量、参数和声明的其他内部函数。当其中一个这样的内部函数在包含它们的外部函数之外被调用时,就会形成闭包。
4.变量的作用域
要理解闭包,首先要理解变量的作用域。
变量的作用域无非就是两种:全局变量和局部变量。
Javascript语言的特殊之处,就在于函数内部可以直接读取全局变量。
其中内部函数中可以访问外部函数的变量,是因为内部函数的作用域链中包含了外部函数的作用域;
也可以理解为:内部函数的作用范围辐射到了外部函数的作用范围;
这里有一个地方需要注意,函数内部声明变量的时候,一定要使用var命令。如果不用的话,你实际上声明了一个全局变量!
5.闭包的写法和用法
刚开始我没意识到,这样写对象也是一种闭包,今天回头想想,这就是闭包的经典用发啊!
6.使用闭包的注意点
1)由于闭包会使得函数中的变量都被保存在内存中,内存消耗很大,所以不能滥用闭包,否则会造成网页的性能问题,在IE中可能导致内存泄露。解决方法是,在退出函数之前,将不使用的局部变量全部删除。
2)闭包会在父函数外部,改变父函数内部变量的值。所以,如果你把父函数当作对象(object)使用,把闭包当作它的公用方法(Public Method),把内部变量当作它的私有属性(private value),这时一定要小心,不要随便改变父函数内部变量的值。