一、闭包(Closure)

1.1、什么是闭包?

理解闭包概念:

a、闭包是指有权限访问另一个函数作用域的变量的函数,创建闭包的常见方式就是在一个函数内部创建另一个函数,也就是创建一个内部函数,创建一个闭包环境,让返回的这个内部函数保存要引用的变量,以便在后续执行时可以保持对这个变量的引用。

b、只要存在调用内部函数的可能,JavaScript就需要保留被引用的函数。而且JavaScript运行时需要跟踪引用这个内部函数的所有变量,直到最后一个变量废弃,JavaScript的垃圾收集器才能释放相应的内存空间。

c、Javascript语言特有的"链式作用域"结构(chain scope),子对象会一级一级地向上寻找所有父对象的变量。

d、如果一个内部函数被调用且引用了它的外部变量那么它就是一个闭包。

相信你看了上面的这段话可能还不理解什么是闭包,那么我就举一个闭包的经典例子来帮助你理解闭包的概念吧。

请看下面这段代码:

<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>闭包</title>
</head>
<body>
<script type="text/javascript">
function out() {
var i = 0;
function inner() {
alert(++i);
}
return inner;
}
var ref= out();
ref();
</script>
</body>
</html>

结果:

上面的代码有两个特点:
1、创建两个函数out,inner,函数inner嵌套在函数out内部,也可以说是函数inner是函数out的内部函数;
2、调用函数out返回内部函数inner。
这样在执行完var ref=out()后,变量ref实际上是指向了函数inner,也可以说是引用了函数inner,再执行ref()后就会看到上图弹出一个窗口显示i的值第一次为1。

这段代码其实就创建了一个闭包,为什么?因为函数out外的变量ref引用了函数out内部的函数inner,也就是说:

当函数out的内部函数inner被函数out外的一个变量ref引用的时候,就创建了一个闭包。

可能你还是不理解闭包这个概念,因为你不知道闭包有什么用,那么先理解一下闭包的作用吧。

1.2、为什么要用闭包(作用)?

1.2.1、保护函数内的变量安全。

解释:以上面的的例子为例,函数out中i只有函数innder才能访问,而无法通过其他途径访问到,因此保护了i的安全性。

1.2.2、通过访问外部变量,闭包可以保存这些变量的上下文环境

在上面的示例中增加了一次函数ref()的调用,执行的结果如下:

解释:当函数out执行完并且最终退出时,它的局部变量会被Javascript的垃圾回收机制回收所占用的资源,也就是局部变量被销毁,但是因为创建了闭包环境,那么内部函数inner就会暂时保存外部函数的局部变量上下文环境。不会被垃圾回收机制回收。所以函数out中的i被复制一份暂时保存下来,这样每次执行ref(),i都是自加1后alert输出i的值。这只是我对闭包作用的简单初浅理解,不专业也不严谨,但大概意思就是这样,理解闭包需要循序渐进的过程。

相信你看了闭包的作用,对理解什么是闭包是否更明白一些,如果还是很疑惑,那么我就再举几个闭包的经典例子来帮助你理解闭包的概念吧。

1.3、闭包的经典示例

1.3.1、示例一

问题:假如我们有如下需求请在页面中放10个div,每个div写上对应的数字,当点击每一个div时显示索引号,如第1个div显示0,第10个div显示9。

可能你会这样做:

<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>闭包</title>
<style type="text/css">
div {
width: 50px;
height: 50px;
background: lightcoral;
float: left;
margin: 20px;
font: 15px/50px "microsoft yahei";
text-align: center;
}
</style>
</head>
<body>
<div>div-1</div>
<div>div-2</div>
<div>div-3</div>
<div>div-4</div>
<div>div-5</div>
<div>div-6</div>
<div>div-7</div>
<div>div-8</div>
<div>dvi-9</div>
<div>div-10</div>
<script type="text/javascript">
var divs=document.getElementsByTagName("div");
for (var i=0;i<divs.length;i++) {
divs[i].onclick=function(){
alert(i);
}
}
</script>
</body>
</html>

结果:

从上面的结果你会发现,不管你点击了哪个div,弹出的框div索引总是10,这可能会让你很意外,会产生疑惑,为什么会出现这样的结果呢?

解释:因为点击事件的函数内部使用外部的变量i一直在变化,当我们指定click事件时并没有保存i的副本,这样做也是为了提高性能,但达不到我们的目的,我们要让他执行的上下文保存i的副本,这种机制就是闭包。

使用闭包可以解决此问题,代码做了如下修改:

<script type="text/javascript">
var div=document.getElementsByTagName("div");
for (var i = 0; i < div.length; i++) {
div[i].onclick=function(n){
              return function(){
alert(n);//产生闭包,引用外部变量。
}
}(i);
}
</script>

结果:

从上面的结果你会发现,使用闭包后,就达到你预期的结果了。

解释:n是外部函数的值,但是内部函数需要使用这个值,因为产生闭包执行环境,所以返回函数前的n被临时驻留在内存中给点击事件使用,简单说就是函数的执行上下文被保存起来,i生成了多个副本。

1.3.2、示例二

示例代码:

<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title></title>
</head>
<body>
<script type="text/javascript">
function out() {
var i = 10;
return function inner() {
i++;
alert(i);//引用了外部变量,创建了闭包环境
};
}
//此处为函数调用,第一个括符为调用out方法,第二个括符为调用返回的inner方法。
out()();
</script>
</body>
</html>

运行 结果:

1.3.3、示例三

示例代码:

<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>闭包</title>
</head>
<body>
<script type="text/javascript">
var name = "this is Window";  
var obj = {    
name: "My Object",
getNameFunc: function() {  
alert(this.name);//输出My Object
return function() {        
return this.name;      
};    
}  
};  
var funn=obj.getNameFunc();
alert(funn());//输出this is Window
</script>
</body>
</html>

运行结果:

解释:funn是属于方法调用,所以this绑定到了obj对象,自然this.name就是"My Object",但是闭包函数无法访问这个this,它只能访问到全局的this,所以this.name就是“this is Window”.

总结:相信通过以上是几个闭包示例,你对闭包也有一定的理解了吧。限于本人才疏学浅,对闭包的理解也并不是很透彻,只是理解了一些表面,会使用而已。若你想理解的更深入推荐你去看stackoverflow这个网站上对闭包的解释:http://stackoverflow.com/questions/111102/how-do-javascript-closures-work

JavaScript学习总结——我所理解的JavaScript闭包的更多相关文章

  1. web前端学习(四)JavaScript学习笔记部分(1)-- JavaScript基础教程

    1.JavaScript基础教程 1.1.Javascript基础-介绍.实现.输出 1.1.1.JavaScript是互联网上最流行的脚本语言,这门语言可用于web和HTML,更可广泛用于服务端.p ...

  2. javascript学习(1)用户的Javascript 放在哪里和函数的绑定方式

    一.实验 1:js脚本放在那里最合适? 1.代码 1.1.test.html <!DOCTYPE html><html>    <head>        < ...

  3. web前端学习(四)JavaScript学习笔记部分(10)-- JavaScript正则表达式

    1.JavaScript正则表达式课程概要 方便查找字符串.数字.特殊字串等等 2.正则表达式的介绍 RegExp是正则表达式的缩写 当检索某个文本时,可以使用一种模式来描述要检索的内容.RegExp ...

  4. web前端学习(四)JavaScript学习笔记部分(8)-- JavaScript 浏览器对象

    1.window对象 1.1.window对象: window对象是BOM的核心,window对象指当前的浏览器窗口 所有javaScript全局对象.函数以及变量均自动生成为window对象的成员 ...

  5. web前端学习(四)JavaScript学习笔记部分(3)-- JavaScript函数+异常处理+事件处理

    1.Javascript函数-了解函数的用途 1.1.函数: 函数是由事件驱动的或者当它被调用时执行的可重复使用的代码块 2.Javascript函数-定义函数 2.1.function必须小写 3. ...

  6. web前端学习(四)JavaScript学习笔记部分(2)-- JavaScript语法详解

    2.1.Javascript语法-运算符(1) 复数运算符 %取余 ++ -- 赋值运算符 += -= *= /= %= 字符串操作 <!DOCTYPE html> <html la ...

  7. web前端学习(四)JavaScript学习笔记部分(9)-- JavaScript面向对象详解

    1.认识面向对象 1.1.概念 1.一切事物皆是对象 2.对象具有封装和继承特性 3.信息隐藏(类的信息隐藏,包括属性和方法) <!DOCTYPE html> <html lang= ...

  8. web前端学习(四)JavaScript学习笔记部分(8)-- JavaScript瀑布流

    index.html <!DOCTYPE html> <html> <head> <meta http-equiv="Content-Type&qu ...

  9. web前端学习(四)JavaScript学习笔记部分(7)-- JavaScript DOM对象控制HTML元素详解

    1.方法 getElementsByName() 获取name 可以获取一个数组类型数据(参数加引号) getElementsByTagName() 获取元素   getAttribute() 获取元 ...

随机推荐

  1. BPM配置故事之案例1-配置简单流程

    某天,Boss找到了信息部工程师小明. Boss:咱们新上了H3 BPM,你研究研究把现在的采购申请流程加上去吧,这是采购申请单. 小明:好嘞 采购申请单 小明回去后拿着表单想了想,开始着手配置. 他 ...

  2. iOS之判断手机号码、邮箱格式是否正确

    //判断手机号码格式是否正确 + (BOOL)valiMobile:(NSString *)mobile{     mobile = [mobile stringByReplacingOccurren ...

  3. 深入理解 Android 之 View 的绘制流程

    概述 本篇文章会从源码(基于Android 6.0)角度分析Android中View的绘制流程,侧重于对整体流程的分析,对一些难以理解的点加以重点阐述,目的是把View绘制的整个流程把握好,而对于特定 ...

  4. CocoaPods的安装、使用、以及遇到的问题

    CocoaPods是什么? 当你开发iOS应用时,会经常使用到很多第三方开源类库,比如JSONKit,AFNetWorking等等.可能某个类库又用到其他类库,所以要使用它,必须得另外下载其他类库,而 ...

  5. PLSql Oracle配置

    1.安装Oracle客户端或者服务端 2.配置环境变量 <1>.一般如果安装了Oracle客户端或者服务端的话,在环境变种的Path中有Oracle的安装路径(计算机-属性-高级系统设置- ...

  6. 【腾讯Bugly干货分享】OCS——史上最疯狂的iOS动态化方案

    本文来自于腾讯Bugly公众号(weixinBugly),未经作者同意,请勿转载,原文地址:https://mp.weixin.qq.com/s/zctwM2Wf8c6_sxT_0yZvXg 导语 在 ...

  7. MemCache超详细解读

    MemCache是什么 MemCache是一个自由.源码开放.高性能.分布式的分布式内存对象缓存系统,用于动态Web应用以减轻数据库的负载.它通过在内存中缓存数据和对象来减少读取数据库的次数,从而提高 ...

  8. 是时候搁置Grunt,耍一耍gulp了

    也算是用了半年Grunt,几个月前也写过一篇它的入门文章(点此查看),不得不说它是前端项目的一个得力助手.不过技术工具跟语言一样日新月异,总会有更好用的新的东西把旧的拍死在沙滩上(当然Grunt肯定没 ...

  9. Thinking in Unity3D:基于物理着色(PBS)的材质系统

    关于<Thinking in Unity3D> 笔者在研究和使用Unity3D的过程中,获得了一些Unity3D方面的信息,同时也感叹Unity3D设计之精妙.不得不说,笔者最近几年的引擎 ...

  10. Azure 部署 Asp.NET Core Web App

    在云计算大行其道的时代,当你在部署一个网站时,第一选择肯定是各式各样的云端服务.那么究竟使用什么样的云端服务才能够以最快捷的方式部署一个 ASP.NET Core 的网站呢?Azure 的 Web A ...