js闭包

作者: 杭州程序员小陈 | 来源:发表于2019-02-19 09:33 被阅读0次

    一、变量的作用域

    要懂得闭包,起首必须懂得Javascript特别的变量作用域。

    变量的作用域无非就是两种:全局变量和局部变量。

    Javascript说话的特别之处,就在于函数内部可以直接读取全局变量。

    Js代码

    ···

    var n=999;

    function f1(){

    alert(n);

    }

    f1(); // 999

    ···

    另一方面,在函数外部天然无法读取函数内的局部变量。

    Js代码

    function f1(){

    var n=999;

    }

    alert(n); // error

    这里有一个处所须要重视,函数内部声明变量的时候,必然要用var。若是不用的话,你实际上声明了一个全局变量!

    Js代码

    function f1(){

    n=999;

    }

    f1();

    alert(n); // 999

    --------------------------------------------------------------------------------------------------------

    二、如何从外部读取局部变量?

    出于各种原因,我们有时要获得函数内的局部变量。然则,前面已经说过了,正常情况下,这是办不到的,只有经由过程变通才能实现。

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

    Js代码

    function f1(){

    n=999;

    function f2(){

    alert(n); // 999

    }

    }

    在上方的代码中,函数f2就被包含在函数f1内部,这时f1内部的所有局部变量,对f2都是可见的。然则反过来就不可,f2内部的局部变量,对f1 就是不成见的。这就是Javascript说话特有的“链式作用域”布局(chain scope),

    子对象会一级一级地向上寻找所有父对象的变量。所以,父对象的所有变量,对子对象都是可见的,反之则不成立。

    既然f2可以读取f1中的局部变量,那么只要把f2作为返回值,我们不就可以在f1外部读取它的内部变量了吗!

    Js代码

    function f1(){

    n=999;

    function f2(){

    alert(n);

    }

    return f2;

    }

    var result=f1();

    result(); // 999

    --------------------------------------------------------------------------------------------------------

    三、闭包的概念

    上一节代码中的f2函数,就是闭包。

    各类专业文献上的“闭包”(closure)定义很是抽象,很丢脸懂。我的懂得是,闭包就是可以或许读取其他函数内部变量的函数。

    因为在Javascript说话中,只有函数内部的子函数才干读取局部变量,是以可以把闭包简单懂得成“定义在一个函数内部的函数”。

    所以,在本质上,闭包就是将函数内部和函数外部连接起来的一座桥梁。

    --------------------------------------------------------------------------------------------------------b

    四、闭包的用处

    闭包可以用在很多处所。它的最大用处有两个,一个是前面提到的可以读取函数内部的变量,另一个就是让这些变量的值始终对峙在内存中。

    怎么来懂得这句话呢?请看下面的代码。

    Js代码

    function f1(){

    var n=999;

    nAdd=function(){n+=1}

    function f2(){

    alert(n);

    }

    return f2;

    }

    var result=f1();

    result(); // 999

    nAdd();

    result(); // 1000

    在这段代码中,result实际上就是闭包f2函数。它一共运行了两次,第一次的值是999,第二次的值是1000。这就说明,函数f1中的局部变量n一向保存在内存中,并没有在f1调用后被主动清除。

    为什么会这样呢?原因就在于f1是f2的父函数,而f2被赋给了一个全局变量,这导致f2始终在内存中,而f2的存在依附于f1,是以f1也始终在内存中,不会在调用停止后,被垃圾收受接管机制(garbage collection)收受接管。

    这段代码中另一个值得重视的一处,就是“nAdd=function(){n+=1}”这一行,起首在nAdd前面没有应用var关键字,是以 nAdd是一个全局变量,而不是局部变量。其次,nAdd的值是一个匿名函数(anonymous function),而这个

    匿名函数本身也是一个闭包,所以nAdd相当于是一个setter,可以在函数外部对函数内部的局部变量进行操纵。

    --------------------------------------------------------------------------------------------------------

    五、应用闭包的重视点

    1)因为闭包会使得函数中的变量都被保存在内存中,内存消费很大,所以不要滥用闭包,不然会造成网页的性能题目,在IE中可能导致内存泄漏。解决办法是,在退出函数之前,将不应用的局部变量全部删除。

    2)闭包会在父函数外部,改变父函数内部变量的值。所以,若是你把父函数当做难象(object)应用,把闭包算作它的公用办法(Public Method),把内部变量算作它的私有属性(private value),这时必然要警惕,不要随便

    改变父函数内部变量的值。

    有权访问另一个函数作用域内变量的函数都是闭包。

    什么是闭包?

    先看一段代码:

    function a(){

    var n = 0;

    function inc() {

    n++;

    console.log(n);

    }

    inc();

    inc();

    }

    a(); //控制台输出1,再输出2

    简单吧。再来看一段代码:

    function a(){

    var n = 0;

    this.inc = function () {

    n++;

    console.log(n);

    };

    }

    var c = new a();

    c.inc(); //控制台输出1

    c.inc(); //控制台输出2

    简单吧。

    什么是闭包?这就是闭包!

    有权访问另一个函数作用域内变量的函数都是闭包。

    这里 inc 函数访问了构造函数 a 里面的变量 n,所以形成了一个闭包。

    再来看一段代码:

    function a(){

    var n = 0;

    function inc(){

    n++;

    console.log(n);

    }

    return inc;

    }

    var c = a();

    c(); //控制台输出1

    c(); //控制台输出2

    看看是怎么执行的:

    var c = couter(),这一句 couter()返回的是函数 inc,那这句等同于 var c = inc;

    c(),这一句等同于 inc(); 注意,函数名只是一个标识(指向函数的指针),而()才是执行函数。

    后面三句翻译过来就是: var c = inc; inc(); inc();,跟第一段代码有区别吗? 没有。

    什么是闭包?这就是闭包!

    所有的教科书教程上都喜欢用最后一段来说明闭包,但我觉得这将问题复杂化了。这里面返回的是函数名,没看过谭浩强C/C++程序设计的同学可能一下子没反应出带不带()的区别,也就是说这种写法自带一个陷阱。虽然这种写法更显高大上,但我还是喜欢将问题单一化,看看代码 1 和代码 2,你还会纠结函数的调用,你会纠结 n 的值吗?

    为啥要这样写?

    我们知道,js的每个函数都是一个个小黑屋,它可以获取外界信息,但是外界却无法直接看到里面的内容。将变量 n 放进小黑屋里,除了 inc 函数之外,没有其他办法能接触到变量 n,而且在函数 a 外定义同名的变量 n 也是互不影响的,这就是所谓的增强“封装性”。

    而之所以要用 return 返回函数标识 inc,是因为在 a 函数外部无法直接调用 inc 函数,所以 return inc 与外部联系起来,代码 2 中的 this 也是将 inc 与外部联系起来而已。

    常见的陷阱

    看看这个:

    function createFunctions(){

    var result = new Array();

    for (var i=0; i < 10; i++){

    result[i] = function(){

    return i;

    };

    }

    return result;

    }

    var funcs = createFunctions();

    for (var i=0; i < funcs.length; i++){

    console.log(funcsi);

    }

    乍一看,以为输出 0~9 ,万万没想到输出10个10?

    这里的陷阱就是:函数带()才是执行函数! 单纯的一句 var f = function() { alert(‘Hi’); }; 是不会弹窗的,后面接一句 f(); 才会执行函数内部的代码。上面代码翻译一下就是:

    var result = new Array(), i;

    result[0] = function(){ return i; }; //没执行函数,函数内部不变,不能将函数内的i替换!

    result[1] = function(){ return i; }; //没执行函数,函数内部不变,不能将函数内的i替换!

    ...

    result[9] = function(){ return i; }; //没执行函数,函数内部不变,不能将函数内的i替换!

    i = 10;

    funcs = result;

    result = null;

    console.log(i); // funcs0就是执行 return i 语句,就是返回10

    console.log(i); // funcs1就是执行 return i 语句,就是返回10

    ...

    console.log(i); // funcs9就是执行 return i 语句,就是返回10

    为什么只垃圾回收了 result,但却不收了 i 呢? 因为 i 还在被 function 引用着啊。好比一个餐厅,盘子总是有限的,所以服务员会去巡台回收空盘子,但还装着菜的盘子他怎么敢收? 当然,你自己手动倒掉了盘子里面的菜(=null),那盘子就会被收走了,这就是所谓的内存回收机制。

    至于 i 的值怎么还能保留,其实从文章开头一路读下来,这应该没有什么可以纠结的地方。盘子里面的菜,吃了一块不就应该少一块吗?

    总结一下

    闭包就是一个函数引用另外一个函数的变量,因为变量被引用着所以不会被回收,因此可以用来封装一个私有变量。这是优点也是缺点,不必要的闭包只会徒增内存消耗!另外使用闭包也要注意变量的值是否符合你的要求,因为他就像一个静态私有变量一样。闭包通常会跟很多东西混搭起来,接触多了才能加深理解,这里只是开个头说说基础性的东西。

    相关文章

      网友评论

          本文标题:js闭包

          本文链接:https://www.haomeiwen.com/subject/paxxyqtx.html