美文网首页技术知识面试面向原型的JS
JS闭包大结局(JS闭包系列3)

JS闭包大结局(JS闭包系列3)

作者: 夏夜星语 | 来源:发表于2016-03-30 21:39 被阅读1268次

在上一篇中再谈JS闭包(JS闭包系列2),我详细的介绍了JS中的变量作用域相关的概念,结合第一节关于JS闭包(JS闭包系列1)的入门, 今天就来对“闭包”这个话题做一个总结。这篇文章信息主要来源于曾探写的《javascript设计模式与开发实践》一书。

JS设计模式与开发实践

衔接上一篇,温习一下:我们已经知道:闭包是由于作用域链的机制自然而然形成的。这一节,希望你能带着这句话来体会每一个实例,以加深对闭包的理解。

变量的寿命兼闭包的第一大作用:延长寿命

除了变量的作用域,另外一个和闭包有着亲密关系的就是变量的生存周期了。一般来说,全局变量的生存周期是永久的,直到我们主动销毁。而在函数内不用var关键字声明的局部变量来说,当退出函数时,这些函数变量立即失去它们的价值,也就被垃圾回收机制销毁了,也算寿终正寝。可是在闭包中,却不是这样。
继续还是以代码说话:

var func = function(){
      var a = 1;  //退出后函数局部变量a直接被销毁
      a++;
      console.log(a);
}; 
func();  //2
func();  //2
func();  //依然是2
普通情况直接销毁

现在看看这段代码:

var func = function(){
      var a = 1;
      return function(){  //匿名函数
            a++;
            console.log(a);
      }
};
var f = func(); //f是对func()的引用
f();   //输出2
f();   //输出3
f();   //输出4
f();   //输出5
闭包封闭变量
由此可见,当退出函数后,局部变量a并没有立即消失,一直存在,这样在第二次调用时a才会是在 2的基础上加1,是3,以后每次调用也才会不断加1;这说明局部变量a一直存活着,寿命延长了!为什么呢?如果你看过我的上一篇文章,你就应该知道,在函数外边是不能访问函数(围墙)里面的变量的,而在这里,f返回了一个匿名函数的引用,那f就可以访问到func()被调用时产生的环境,也就是func()的生存空间,f可以进去闲逛啦!那么想象一下,假如你是这个名叫"func"的大观园的主人,你能不让随时都可能来的“刘姥姥”进去看看吗?既然让进,那么里面的亭台楼榭,一花一石恐怕都不能在园子一盖好,就把它们销毁了吧?

闭包的第二大作用:封闭变量

那么既然闭包可以有这样一个机制,我们可以用它来干什么呢?下面就来介绍介绍闭包这个特殊角色的奇技淫巧。其实上面已经有了第一大作用:延长寿命!现在来看一下他的第二个作用:封闭变量。其实这个也很好理解,闭包闭包,从字面上都可以理解有封闭作用啦。
继续例子:

<html>
    <body>
        <div>1</div>
        <div>2</div>
        <div>3</div>
        <div>4</div>
        <div>5</div>
    <script>
        var nodes = document.getElementsByTagName('div');
        for(var i = 0, len = nodes.length; i < len; i++){
            nodes[i].onclick = function(){
                alert(i);
            }
        };
    </script>
  </body>
</html>

测试这段代码,就会发现无论点击那个div,最后弹出的结果都是5.这是因为div节点的onclick事件是被异步触发的,当事件被触发的时候,for循环早已经结束,所以i变量的值已经是5,在后续onclick事件查找时肯定就是5啦。

那么怎么解决才能让div返回如我们所愿?这时就是闭包大展身手的时候啦!思路是将每次循环的i值封闭起来, 当沿着作用域链从内到外查找变量i时,会先找到被封闭在闭包环境中的i,这样的话,如果有5个div,这里的i就分别是0,1,2,3,4啦:

for(var i = 0, len = nodes.length; i < len; i++){
    (function(i){       
        nodes[i].onclick = function(){
                console.log(i);
            }
        })(i)
};

第三大作用:模拟面向对象

来看下面用面向对象实现的代码:

var extent = {
    value:0,
    call:function(){
        this.value++;
        console.log(this.value);
    }
};
extent.call();  //输出:1
extent.call();  //输出:2
extent.call();  //输出:3

或者:

var Extent = function(){
    this.value = 0;
};
Extent.prototype.call = function(){
    this.value++;
    console.log(this.value);
};
var extent = new Extent();
extent.call(); //输出:1
extent.call(); //输出:2
extent.call(); //输出:3

如果用闭包的方法,该怎么写呢?

var extent = function(){
    var value = 0;
    return {
        call:function(){
            value++;
            console.log(value);
        }
    }
};
var extent = extent();
extent.call();  //输出:1
extent.call();  //输出:2
extent.call();  //输出:3

领略了这么多的奇技淫巧,是不是对JavaScript的闭包有了更深的理解?其实闭包的用途远不止这么多,更精彩的还需要大家在实践中多多发现。由于作者水平有限,暂时只额能给大家分享到这里,希望后续可以继续了解,在实践中不断学习,成长。

本篇完

相关文章

  • JS闭包大结局(JS闭包系列3)

    在上一篇中再谈JS闭包(JS闭包系列2),我详细的介绍了JS中的变量作用域相关的概念,结合第一节关于JS闭包(JS...

  • php之闭包函数(Closure)

    php闭包函数(Closure) JS闭包 js和php闭包使用和区别

  • JS闭包

    JS闭包 闭包练习

  • 再谈JS闭包(JS闭包系列2)

    这篇文章,来继续谈谈Javascript闭包的剩余问题。因为在上一篇文章中关于JS闭包(JS闭包系列1)主要简单的...

  • JS闭包问题(二)

    在之前的JS闭包问题(一)文章中大概介绍了一下JS闭包,同时讲了闭包与变量之间的问题,今天我们继续聊闭包,聊聊闭包...

  • 关于JS闭包(JS闭包系列1)

    今天做笔试题,好多关于JS闭包问题,就整理出来,分享一下。 首先,引用下所谓的“官方解释”:所谓“闭包”,指的是一...

  • JS闭包入门

    最近有看到朋友的面经里提到的JS闭包的问题,就想研究研究,以下是我对JS闭包的简单理解。 到底什么是JS闭包? 定...

  • 学习JavaScript闭包和作用域笔记

    JS JavaScript闭包和作用域 闭包 JavaScript高级程序设计中对闭包的定义:闭包是指有权访问另外...

  • 简单的聊一下闭包

    js中的闭包 闭包是学习js中永远也绕不过去的一个坎,那么,今天我们就去一段简单的代码开始聊一聊闭包 什么是闭包 ...

  • 浅谈闭包

    js中的闭包 闭包是学习js中永远也绕不过去的一个坎,那么,今天我们就去一段简单的代码开始聊一聊闭包 什么是闭包 ...

网友评论

  • xiaoaiai:我能说你写的文章很有帮助吗 完美理解闭包 作用域
    夏夜星语:@xiaoaiai 能对别人有帮助是很令我幸福的事,谢谢
  • fangkyi03:把this作用域去理解成原型链就可以了 函数本身就是一个object类型的 只不过跟原型链区别的在于 原型链所有东西需要自己定义 而function的话 在执行的时候就把里面的所有声明跟赋值都变成了{aaa:xxx,xxx:()}这种形式 并且继承了外部的全局作用域 函数作用域 块级作用域里面的所有东西 改成了从scope中进行获取 但是回过头来看 其实跟原型链可以说一模一样 没有半点区别 所以从原型链的角度去理解this作用域跟闭包是更加好的 不然会被绕进去
  • 38faf480efd2:有一个点不是很明白,(function(){})
    这种写法,我看很多人使用()()就可以调用了,但是两个括号并不能知道是这个函数呢?
    你是怎么理解的
  • smartphp:顶,这本书确实很好。
  • a4a8de24c022:<html>
    <body>
    <div>1</div>
    <div>2</div>
    <div>3</div>
    <div>4</div>
    <div>5</div>
    <script>
    var nodes = document.getElementsByTagName('div');
    for(var i = 0, len = nodes.length; i < len; i++){
    nodes[i].onclick = function(){
    alert(i);
    }
    };
    </script>
    </body>
    </html>
    小白求问,为啥这个例子我没看懂,点击事件是异步执行的,在点击之前,for循环已经执行完,所以此时的i为5了,但是之前每一次的for循环,都有对点击事件进行赋值 nodes[0].onclick = function() { alert(0); } nodes[1].onclick = function() { alert(1); },此时的 i=5 是怎样影响到点击事件的?

  • a4a8de24c022:<html>
    <body>
    <div>1</div>
    <div>2</div>
    <div>3</div>
    <div>4</div>
    <div>5</div>
    <script>
    var nodes = document.getElementsByTagName('div');
    for(var i = 0, len = nodes.length; i < len; i++){
    nodes[i].onclick = function(){
    alert(i);
    }
    };
    </script>
    </body>
    </html>
    小白求问,为啥这个例子我没看懂,点击事件是异步执行的,在点击之前,for循环已经执行完,nodes[0].onclick = function(){ alert(0);},nodes[1].onclick = function(){ alert(1);}...,点击事件已经都赋值过一遍了吧,为啥alert的都是5
    fangkyi03:@吃饭睡觉玩简书 因为这里没有使用let的关系 所以var在执行的时候将当前作用域的i值进行了替换 所以当点击时间被改变的时候 引用的是最新的5
    840a80d9a3bb: @吃饭睡觉玩简书 node[i]中i和alert(i)中的i不是同一个!他们之间不相等
  • 2f4033260f75:最后一个例子,在console面板打印没问题,IDE里自己写出来调试就报错,extent没有被定义~
  • MonaSong:我去调试了一下代码,发现用闭包的方式写时,for循环的时候实际上把i值都一一赋值了,当每个点击的时候就会把之前存储好的i值直接拿出来用
    夏夜星语:@安静的心情 嗯,在用闭包实现的时候,是会保持值的。

本文标题:JS闭包大结局(JS闭包系列3)

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