本文采用5W1H分析法来看一下闭包。

一、why-----从为什么要引入闭包先来了解一下闭包。

讨论为什么要引入闭包就要先讨论一下js中的作用域链及垃圾回收机制。

熟悉js语言的人都知道js中有作用域的概念和垃圾回收机制。那么我们首先来看一下js中的作用域链

  1. 作用域链:

js中的变量执行环境分为全局执行环境和局部执行环境。当代码在执行时会创建变量对象的一个作用域链。而作用域链简单来说,就是函数在定义的时候创建的上下文执行环境,用于在标识符解析中变量查找。作用域链的前端始终都是当前执行代码所在环境的变量对象即函数自身的本地变量,作用域链的下一个变量对象来自包含环境即父级函数,而在下一个变量对象则来自下一个包含环境,然后一直延续到全局执行环境;全局执行环境的变量对象始终是作用域链的最后一个对象。js中标识符解析就是沿着作用域链一级一级地搜索标识符的过程;搜索过程从作用域链的前端开始逐级向上搜索,直到找到标识符为止。

作用域链只能从下往上查找而不能从上往下查找,举例来说函数内部可以访问到全局变量,而在函数外部不能访问函数内部变量。如下代码:

var n=999;
function f1(){
  alert(n);
}
f1(); //

代码中n为全局变量,而在函数内部可以直接访问到。再看以下代码:

function f1(){
   var n=999;
}
alert(n); // 会报错 n is not defined

当在函数外部访问内部变量时会报错。了解了作用域链再来看一下js垃圾回收机制。

2.js中垃圾回收机制:

javascript时一门具有自动垃圾回收机制的语言。一般来说,一个函数在执行开始的时候,会给其中定义的局部变量划分内存空间以便存储他们的值,然后在函数中使用变量,等到函数执行完毕返回了,局部变量就没有存在的必要了,这些变量就被认为是无用的了;因此释放他们的内存供将来使用,下次再执行此函数的时候,所有的变量又回到最初的状态,重新赋值使用。由于作用域链和垃圾回收机制的限制,若需要在函数外部访问函数内部的变量则访问不到。若要想在函数外部访问函数内部的变量怎么访问呢?此时闭包就华丽登场了(*^_^*)。

二、what-----什么是闭包呢从概念来理解

  《javascript高级程序设计第三版》中是这样定义闭包的“闭包是指有权访问另一个函数作用域中的变量的函数”。所有的函数都可以说是闭包。在我理解,简单来说就是在一个函数内部嵌套一个函数,内部函数可以引用外部函数的参数和变量,参数和变量不会被回收,使局部变量能够在全局中访问到,暴露局部变量能够让他的上一层环境访问到。闭包由函数和创建该函数的执行环境构成。当在函数内部嵌套另外一个函数时,并且外部函数将嵌套的函数对象作为返回值返回就是闭包的一种。如下代码实例:

function aaa() {
var a=1;
function bbb(){
a++;
alert(a);
}
return bbb;
} var c=aaa();
c();//2

内部函数bbb可以访问到外部函数中的a变量。当外部函数执行完后,由于内部函数引用了外部函数的变量,因此内部函数的作用域包含了外部函数的活动对象,当外部函数被调用返回后其执行环境被销毁,但内部函数的作用域链依然存在,因其内部函数还未被销毁所以其外部函数的活动对象仍然在内存中,如在bbb中访问a变量依然可以访问到。等其内部函数执行完后作用域链被销毁,释放变量内存。了解了闭包的概念,再来看一下闭包的应用场景。

三、when------什么时候使用闭包?

当希望一个变量长期驻扎在内存中时,此时使用闭包可以使函数执行完后其活动对象仍然在内存中。

四、where----闭包应用在哪?

闭包好处:1、避免全局变量的污染
2、可以模仿块级作用域
3、私有成员的存在

1、闭包可以用来模仿块级作用域。

js中本身没有块级作用域的概念,而使用闭包可以用来模仿块级作用域。

function outputNumbers(count){
(function(){
for(var i=0;i<count;i++){
alert(i);
}
})();
alert(i);//报错 i is not defined
}

在for循环外部插入一个私有作用域,变量i只能在循环中使用,使用后即被销毁。所以在私有作用域外部访问不到。

2、在循环中直接找到对应的索引。

若页面中有多个li标签,而需要给每个标签添加一个事件,则可以这样做:

for(var i=0;i<aLi.length;i++){
(function(i){
aLi[i].onclick=function(){
alert(i);
}
})(i);
}

如果不使用闭包的话,那么弹出的i为aLi.length的值。

for(var i=0;i<6;i++){
aLi[i].onclick=function(){
alert(i);//
}
}

3、用闭包模拟私有方法:

js中本身没有私有成员的概念,所有对象属性都是公有的。闭包可以用来创建私有变量。如下代码:

var aaa = (function(){
var a = 1;
function bbb(){
a++;
alert(a);
}
function ccc(){
a++;
alert(a);
}
return {
b:bbb, //json结构
c:ccc
}
})();
aaa.b(); //
aaa.c()

4、模块化代码,避免全局变量的污染:

var abc = (function(){      //abc为外部匿名函数的返回值
var a = 1;
return function(){
a++;
alert(a);
}
})();
abc(); //2 调用一次abc函数,其实是调用里面内部函数的返回值
abc(); //

接下来再来看一下闭包的缺点:闭包会使变量始终保存在内存中,如果不当使用会增大内存消耗。还有由于IEjs对象和DOM对象使用不同的垃圾收集方法,因此闭包在IE中会导致内存泄露问题,也就是无法销毁驻留在内存中的元素。只有关闭浏览器时才会释放内存。如下代码:

function fn(){
var oDiv = document.getElementById('div1');//oDiv用完之后一直驻留在内存中
oDiv.onclick = function () {
alert(oDiv.id);//这里用oDiv导致内存泄露
};
}
fn();
//最后应将oDiv解除引用来避免内存泄露
function fn(){
var oDiv = document.getElementById('div1');
oDiv.onclick = function () {
alert(oDiv.id);
};
oDiv = null;
}

由于oDiv.onclick中引用了oDiv.id,互相引用时存在内存泄漏,因此需手动解除引用来避免内存泄漏。

js之认识闭包的更多相关文章

  1. 详解js中的闭包

    前言 在js中,闭包是一个很重要又相当不容易完全理解的要点,网上关于讲解闭包的文章非常多,但是并不是非常容易读懂,在这里以<javascript高级程序设计>里面的理论为基础.用拆分的方式 ...

  2. js中的闭包之我理解

    闭包是一个比较抽象的概念,尤其是对js新手来说.书上的解释实在是比较晦涩,对我来说也是一样. 但是他也是js能力提升中无法绕过的一环,几乎每次面试必问的问题,因为在回答的时候.你的答案的深度,对术语的 ...

  3. Js(javaScript)的闭包原理

    问题?什么是js(javaScript)的闭包原理,有什么作用? 一.定义 官方解释:闭包是一个拥有许多变量和绑定了这些变量的环境的表达式(通常是一个函数),因而这些变量也是该表达式的一部分.  小编 ...

  4. 浅谈JS中的闭包

    浅谈JS中的闭包 在介绍闭包之前,我先介绍点JS的基础知识,下面的基础知识会充分的帮助你理解闭包.那么接下来先看下变量的作用域. 变量的作用域 变量共有两种,一种为全局变量,一种为局部变量.那么全局变 ...

  5. js中的“闭包”

    js中的“闭包” 姓名:闭包 官方概念:闭包是一个拥有许多变量和绑定了这些变量的环境的表达式(通常是一个函数),因而这些变量也是该表达式的一部分. ( ⊙o⊙ )!!!这个也太尼玛官方了撒,作为菜鸟的 ...

  6. JS中的闭包(closure)

    JS中的闭包(closure) 闭包(closure)是Javascript语言的一个难点,也是它的特色,很多高级应用都要依靠闭包实现.下面就是我的学习笔记,对于Javascript初学者应该是很有用 ...

  7. Js中的闭包原理

    要了解清楚js中的闭包制机,那么得先了解全局执行环境.块级执行环境.函数执行环境.变量对象.环境栈.作用域链.摧毁执行环境. 全局执行环境 全局执行环境指的是最外层的执行环境.在web中全局执行环境被 ...

  8. js中的闭包理解一

    闭包是一个比较抽象的概念,尤其是对js新手来说.书上的解释实在是比较晦涩,对我来说也是一样. 但是他也是js能力提升中无法绕过的一环,几乎每次面试必问的问题,因为在回答的时候.你的答案的深度,对术语的 ...

  9. JS基础学习——闭包

    JS基础学习--闭包 什么是闭包 闭包的定义如下,它的意思是闭包使得函数可以记住和访问它的词法范围,即使函数是在它声明的词法范围外执行.更简单来讲,函数为了自己能够正确执行,它对自己的词法范围产生闭包 ...

  10. JS面向对象之闭包

    闭包 闭包的概念 闭包的含义就是闭合,简单来说就是一个具有封闭功能的结构 闭包就是具有封闭的对外不公开的,包裹结构或空间 在 js 中的闭包 在 js 中函数可以构成闭包 函数是一个代码结构的封闭结构 ...

随机推荐

  1. Stream流

    在Node中,存在各式各样不同的数据流,Stream(流)是一个由不同对象实现的抽象接口.例如请求HTTP服务器的request是一个 流,类似于stdout(标准输出):包括文件系统.HTTP 请求 ...

  2. [原]CentOS7.2最小安装环境部署Asp.NET Core笔记

    转载请注明原作者(think8848)和出处(http://think8848.cnblogs.com) 写在前面的话 不知不觉在cnblogs上注册已经10多年了,看我的园龄就直接暴露了我实际年龄, ...

  3. RandHelper

    /// <summary> /// 类说明:RandHelper类,用来实现取随机 /// 编码日期:2015-12-29 /// 编 码 人:里程碑 /// 修改日期:2015-12-2 ...

  4. neo4j-jersey分嵌入式和服务式连接图形数据库

    原文载自:http://blog.csdn.net/yidian815/article/details/12887259 嵌入式: 引入neo4j依赖 <dependency> <g ...

  5. 由于服务器意外的断电,导致SQL SERVER服务器上数据库出现“置疑”而无法使用,

    来自百度 1.停止数据库服务器,将数据库MDF文件和LDF文件复制备份一份2.启动数据库服务器,删除置疑的数据库3.仅用备份的数据库MDF文件附加数据库,sp_attach_db或者sp_attach ...

  6. React.js 官网入门教程 分离文件 操作无法正常显示HelloWord

    对着React官网的教程练习操作,在做到分离文件练习时,按照官网步骤来却怎么也无法正常显示HelloWord. 经测试,html文件中内容改为: <!DOCTYPE html><ht ...

  7. chorme浏览器调试Android设备

    Android设备开启开发者模式,并打开USB调试: 接着在Android设备上运行项目 在chrome浏览器打开F12: 在Remote devices里即可调试页面. ! 一般需要FQ

  8. Beta阶段第九次Scrum Meeting

    情况简述 BETA阶段第九次Scrum Meeting 敏捷开发起始时间 2017/1/2 00:00 敏捷开发终止时间 2017/1/3 00:00 会议基本内容摘要 deadline临近 参与讨论 ...

  9. 浅谈:javascript的面向对象编程之基础知识的介绍

    在进入javascript的面对对象之前,我们先来介绍一下javascript的几个概念. 1.javascript的面向对象的基本概念 function aa(){ } /* * 这里的aa,在我们 ...

  10. 一个用纯CSS实现的下拉菜单

    用CSS做下拉列表的特效,代码如下: <html xmlns="http://www.w3.org/1999/xhtml" lang="en"> & ...