js闭包问题

js的闭包

1.变量的作用域

要理解闭包,首先必须理解javascript特殊的变量作用域。
变量的作用域无非就是两种:全局变量和局部变量。
javascript语言的特殊之处,就是在于函数内部可以直接读取全局变量。

举例来说

//内部访问全局变量
var n = 999;
function f1(){
    alert(n);
}
f1();//999
//另一方面,在函数外部无法读取函数内部的局部变量
function f1(){
    var n = 123;
}

alert(n);//n is not defined

上述代码验证了js里变量的作用域
这里有一个地方需要注意,函数内部声明变量的时候,一定要使用var 命令。如果不用的话,你实际上声明了一个全局变量!!!!


2.那么应该如何从外部读取局部变量呢?

出于各种各样的原因,我们有时候需要得到函数内部的局部变量。但是前面已经说过了,正常情况是不可能的,所以只能做一些小手段。

那就是函数内部在定义一个函数。

function f1(){
    n = 123;
    function f2(){
        console.log(n);//123    
    }
}

在上面的代码中,函数f2就被包括在函数f1内部,这时f1内部的所有全局变量,对f2都是可见的。但是反过来就不行,f2内部的局部变量,对f1就是不可见的。
这就是JavaScript语言特有的“链式作用域”结构,子对象会一级一级地向上寻找所有父对象的变量。
所以,父对象的所有变量,对子对象都是可见的,反之则不成立。
既然f2可以读取f1中的局变量,那么只要把f2作为返回值,我们不就可以再f1外部读取它的内部变量了吗!

function f1(){
    n = 123;
    (function f2(){
        n++;
        return n;           
    })()
    return n;
}
var a = f1();
console.log(a);//124

f2自调用了一下。就是f2 将n加一后,return出来,f1也将n的值return出来。

将f1()执行一次然后将n的值复制给a 最后打印出来了124。


3.闭包的概念

上一节代码f2的函数,就是闭包。
我借用别人的理解的闭包的概念就是,能够读取其它函数内部变量的函数。

由于JavaScript语言中,只有函数内部的的子函数才能读取局部变量,因此可以把闭包简单理解成“定义在一个函数内部的函数”

所以,闭包的意义就是讲函数内部和外部链接起来的意思。


4.闭包的用途

闭包是可以用在很多的场景当中的。它的最大的用处有两个,

  1. 一个是前面提到的可以读取函数内部的变量;
  2. 另一个就是让这些变量的值始终保持在内存中;
function f1(){
    var n = 123;
    nAdd = function(){
        n++;    
    }
    function f2(){
        console.log(n)'
    }
    return f2;
}
var result = f1();
result();//123
nAdd();
result();//124

在这段代码中,result实际上就是闭包f2函数。它一共运行了两次,第一次的值是123,第二次的值是1234。
这就证明了,函数f1中的变量n一直保存在内存中,并没有在f1调用后被垃圾回收机制给回收。
主要原因就是f1是f2的复函数,而f2被赋给了一个全局变量,这导致f2始终在内存中,而f2的存在依赖于f1,因此f1也始终在内存中,不会在调用结束后,被垃圾回收机制回收。


5.使用闭包的注意点

  1. 由于闭包会使得函数中的变量都被保存在内存中,内存消耗很大,所以不能滥用闭包,否则会造成网页的性能问题,在ie中可能会造成内存泄漏。
    解决的办法是,在函数退出之前,将不使用的局部变量全部删除。
  2. 闭包会在父函数外部,改变父函数内部变量的值。所以,如果你把父函数当做对象使用,把闭包当做它的公共方法,把内部变量当做它的私有属性,这时一定要小心,不要随便改变父函数内部变量的值。

你可能感兴趣的:(JavaScript学习之旅)