介绍js如何定义变量的及闭包的使用
let var const 之间的区别
var
var的作用域是根据函数作用域的, 他在函数内起作用 这里就引用到闭包和作用域链
1 | function out(){ |
var 的变量提升
1 | function (){ |
let
let的作用域的块级作用域,不会变量提升 可以理解为在某块范围内作用 与var 的函数作用域不同在于 let的作用域会形成闭包
1 | for (var i = 0; i < 3; i++) { |
1 | var会在全局声明 而setTimeout 是一个定时器 也是异步回调 会在同步执行完毕再执行异步 所以当for循环完毕 i=3时调用三次定时器 |
const
const代表一个值的常量索引,常量的值在垃圾回收前永远不能改变,所以需要谨慎使用。
cosnt定义的值不能被改变 否则会报错
1 | Uncaught TypeError: Assignment to constant variable. |
闭包
外部可以访问内部的私有变量
可以返回携带状态的函数,并且状态隐藏 可以做到变量的私有
1 | function f1(){ |
在上面的代码中,函数f2就被包括在函数f1内部,这时f1内部的所有局部变量,对f2都是可见的。但是反过来就不行,f2内部的局部变量,对f1就是不可见的。这就是Javascript语言特有的”链式作用域”结构(chain scope),子对象会一级一级地向上寻找所有父对象的变量。所以,父对象的所有变量,对子对象都是可见的,反之则不成立。
既然f2可以读取f1中的局部变量,那么只要把f2作为返回值,我们不就可以在f1外部读取它的内部变量了吗!
1 | function f1(){ |
闭包就是能够读取其他函数内部变量的函数
要形成闭包 必须两层function函数作用域
1 | function f1(){ |
这里形成闭包后,f1的局部变量n一直保存在内存中,并没有因为f1的重复调用而清除
原因就在于f1是f2的父函数,而f2被赋给了一个全局变量,这导致f2始终在内存中,而f2的存在依赖于f1,因此f1也始终在内存中,不会在调用结束后,被垃圾回收机制(garbage collection)回收。
缺点
由于闭包会使得函数中的变量都被保存在内存中,内存消耗很大,所以不能滥用闭包,否则会造成网页的性能问题,在IE中可能导致内存泄露。解决方法是,在退出函数之前,将不使用的局部变量全部删除。
闭包会在父函数外部,改变父函数内部变量的值。所以,如果你把父函数当作对象(object)使用,把闭包当作它的公用方法(Public Method),把内部变量当作它的私有属性(private value),这时一定要小心,不要随便改变父函数内部变量的值。
1 | var name = "The Window"; |
经典闭包题
个人理解
object.getNameFunc() 执行到这一步时 调用object内的函数 返回一个function(){return this.name} 这时候 name还是My Object 当再次执行时object.getNameFunc()() 调用了外部的name 输出为The Window