我印象中,javascript的闭包属于进阶的范畴,无非是用来在面试中装装逼而已。你看我身边的一个小伙子,有一天我装逼地问他什么是javascript的闭包,他居然连听都没听说过。但他做起前端的东西来很快,就像我见过的其他小伙子一样。

这说明,知不知道闭包,似乎无关重要。

但是,这几天写了一些前端代码,觉得还是应该要了解这个东西。

所谓的闭包,按我的理解,就是一个javascript函数(父函数)里面有子函数,然后外界可以通过这个子函数来访问到父函数里面的变量。闭包的作用在于保护这些父函数里的变量,使得这些变量在父函数运行期结束后,继续存活,免遭javascript垃圾回收机制所回收。

怎么理解呢?举例说明。

比如说,我们在页面上输出一堆列表项,每个列表项都有一个onclick事件:

<ul>
<li id="li1" onclick="hi(1)">我是1</li>
<li id="li2" onclick="hi(2)">我是2</li>
...
</ul>

我们会发现,列表项存在着许多重复代码,造成页面代码很多。如果我们不预先给每个

  • 设置事件,而是在页面加载完毕以后,再给它们绑定onclick事件,这样页面就简洁许多,减少带宽的消耗。
  • 那么,如何绑定事件呢?假如我们这样写:

    function hi(number){
    alert("我是 " + number + " !");
    }
    $(function(){
    for(var i = 1;i <= 100; i++){
    $("#li" + i).bind("click",function(){
    hi(i);
    });
    }
    });

    那么最终,点击每个列表项,都会显示 “我是 100 !”。原因是循环结束后,i是100。

    如果换成这样子:

    function hi(number){
    return function(){//子函数,闭包机制将传入的number存储起来
    alert("我是 " + number + " !");
    };
    }
    $(function(){
    for(var i = 1;i <= 100; i++){
    $("#li" + i).bind("click",hi(i));
    }
    });

    那么就可以得到我们想要的结果。原因是,函数hi(number)中,建立了闭包,在事件绑定的时候,将传过来的i储存起来了。

    要理解上述语句,有必要对事件绑定这里做一些说明:

    $("#li" + i).bind("click",函数定义);

    意思是,绑定给元素事件的,是函数定义,而不是一个函数运行语句!一定要清楚这一点。比如说,function hi(numbert){…}是函数定义,hi(i)就是一个函数运行语句。

    我们对代码稍加修改,对比来看一下:

    function hi(number){//函数体内是普通的执行语句
    alert("我是急性子 " + number + " !");
    }
    function hi_bb(number){//函数体内返回一个子函数实例
    return function(){//子函数,闭包
    alert("我是 " + number + " !");
    };
    }
    $(function(){
    for(var i = 1;i < 3; i++){
    //此时hi执行,一次性弹出信息,绑定失败
    $("#li" + i).bind("click",hi(i));
    //此时hi_bb执行,返回了一个函数定义,绑定成功
    $("#li" + i).bind("click",hi_bb(i));
    }
    });

    在事件绑定的时候,函数hi、hi_bb都执行了。hi里面是普通的执行语句,呼啦啦的一下执行完毕,列表项未能绑定事件;而hi_bb执行后,返回的是函数定义,因此列表项事件绑定成功。

    让我们再回顾一下闭包的作用:保护父函数里的变量,使得这些变量在父函数运行期结束后,继续存活,免遭javascript垃圾回收机制所回收。 本文例子中,事件绑定之时,hi_bb执行,i作为参数传入。本来循环语句、页面onload函数结束,i就应该不存在了。但因为hi_bb里有闭包,每一个i的值都保存了下来,在以后的点击事件中,发挥出应有的作用。

    这些话好像有些勉强和装逼,我也是在体会中。如果是面向对象语言,这很好理解,对象构造自同一个类,而对象里的属性、变量的值却是不同的。把参数通过构造函数、成员方法,或者属性设置传给对象的属性或成员变量以后,对象就会将它们保存下来。javascript不是面向对象,没有类,只有函数。也许闭包就是它的一种面向对象的模拟。我是这么理解的。

    我以前也总结过闭包,现在又写了一篇,感觉又加深了一点理解:

    http://blog.csdn.net/leftfist/article/details/41868659

    可以在网站 http://jsfiddle.net/ 上运行一下这些代码,加以印证、对照。

    版权声明:本文为博主原屙文章,喜欢你就担走。

    javascript闭包的应用的更多相关文章

    1. 《Web 前端面试指南》1、JavaScript 闭包深入浅出

      闭包是什么? 闭包是内部函数可以访问外部函数的变量.它可以访问三个作用域:首先可以访问自己的作用域(也就是定义在大括号内的变量),它也能访问外部函数的变量,和它能访问全局变量. 内部函数不仅可以访问外 ...

    2. JavaScript 闭包深入浅出

      闭包是什么? 闭包是内部函数可以访问外部函数的变量.它可以访问三个作用域:首先可以访问自己的作用域(也就是定义在大括号内的变量),它也能访问外部函数的变量,和它能访问全局变量. 内部函数不仅可以访问外 ...

    3. JavaScript闭包(Closure)

      JavaScript闭包(Closure) 本文收集了多本书里对JavaScript闭包(Closure)的解释,或许会对理解闭包有一定帮助. <你不知道的JavsScript> Java ...

    4. Javascript闭包和C#匿名函数对比分析

      C#中引入匿名函数,多少都是受到Javascript的闭包语法和面向函数编程语言的影响.人们发现,在表达式中直接编写函数代码是一种普遍存在的需求,这种语法将比那种必须在某个特定地方定义函数的方式灵活和 ...

    5. javascript闭包理解

      //闭包理解一 function superFun(){ var _super_a='a'; function subfuc(){ console.log(_super_a); } return su ...

    6. Javascript闭包深入解析及实现方法

      1.什么是闭包 闭包,官方对闭包的解释是:一个拥有许多变量和绑定了这些变量的环境的表达式(通常是一个函数),因而这些变量也是该表达式的一部分.闭包的特点:1. 作为一个函数变量的一个引用,当函数返回时 ...

    7. javascript闭包和作用域链

      最近在学习前端知识,看到javascript闭包这里总是云里雾里.于是翻阅了好多资料记录下来本人对闭包的理解. 首先,什么是闭包?看了各位大牛的定义和描述各式各样,我个人认为最容易一种说法: 外部函数 ...

    8. JavaScript闭包深入解析

      for (var i=1; i<=5; i++) { setTimeout( function timer() { console.log( i ); }, i*1000 ); } --上面这段 ...

    9. JavaScript 闭包系列二(匿名函数及函数的闭包)

      一. 匿名函数 1. 函数的定义,可分为三种 1) 函数声明方式 function double(x) {     return 2*x; } 2)Function构造函数,把参数列表和函数体都作为字 ...

    10. JavaScript闭包模型

        JavaScript闭包模型 -----  [原创翻译]2016-09-01  09:32:22 < 一>  闭包并不神秘 本文利用JavaScript代码来阐述闭包,目的是为了使普通 ...

    随机推荐

    1. validate 常用的输入框校验

      记录一下angular可以直接用的输入框校验器,外加一个国内手机号码的校验 <!DOCTYPE html> <html> <head> <meta chars ...

    2. GIL和copy

      GIL: Global Interpreter Lock 全局解释器锁 多任务执行占CPU 多任务占用CPU的资源消耗:进程>线程>协程 在cpython解释器中只有进程是真的多任务,线程 ...

    3. Http中cookie的使用以及用CookieManager管理cookie

      前段时间项目需要做个接口,接口需要先登录才能进行下一步操作,这里就需要把登录的信息携带下去,进行下一步操作.网上查了很多资料,有很多种方法.下面就介绍较常用 的. 第一种方式: 通过获取头信息的方式获 ...

    4. Couchbase III(Python Library)

      Couchbase III(Python Library) 第一步 安装 使用pip安装: >pip install couchbase --quiet 确认是否安装成功: >python ...

    5. 关于对象字面量花括号{} key的类型

      原来对{}方式建立的实例化后的对象的key,理解不清,只知道,数组加不加“”,反正都是字符,现在要理解下,到底怎么回事?返回结果是 var a={}; a[1]="a"; a[&q ...

    6. MySQL语句之or和and多条件查询

      在where中可以包含任意数目的and和or操作符,需要注意的是在没有任何其他符号的时候,例如括号,AND的优先级高于OR,因此,当两者一起使用时,先运算AND两边的条件表达式,再运算OR两边的条件表 ...

    7. 【ITOO 2】.NET 动态建库建表:使用SQL字符串拼接方式

      导读:在最近接手的项目(高效云平台)中,有一个需求是要当企业用户注册时,给其动态的新建一个库和表.刚开始接手的时候,是一点头绪都没有,然后查了一些资料,也问了问上一版本的师哥师姐,终于有了点头绪.目前 ...

    8. windows 环境下.Net使用Redis缓存

      Redis简介 Redis是一个开源的,使用C语言编写,面向“键/值”对类型数据的分布式NoSQL数据库系统,特点是高性能,持久存储,适应高并发的应用场景.Redis纯粹为应用而产生,它是一个高性能的 ...

    9. [luoguP2461] [SDOI2008]递归数列(DP + 矩阵优化)

      传送门 本题主要是构造矩阵,我们只需要把那一段式子看成两个前缀和相减, 然后就直接矩阵连乘. 直接对那个k+1阶矩阵快速幂即可,注意初始化矩阵为单位矩阵,即主对角线(左上到右下)都为1其他都为0. 另 ...

    10. [luoguP2184] 贪婪大陆(树状数组)

      传送门 用两个树状数组,cr 维护 1....x 中 r 的数量 cl 维护 1....x 中 l 的数量 求答案的时候只需要求 y 前面 被作为左端点 的个数 - x 前面 被作为右端点的个数 —— ...