1、闭包的经典错误

假如页面上有若干个div,我们想给它每个绑定一个onclick方法,于是有了下面的代码。

function A(){
var divs=document.getElementsByTagName("div");
for(var i=0; i<divs.length; i++){
divs[i].onclick=function(){
alert(i);
}
}
}

我原以为每次都弹出相对应的i,可是结果却是每次都弹出divs.length的值。于是查资料后知道解决方法是这样的。

function A(){
var divs=document.getElementsByTagName("div");
for(var i=0; i<divs.length; i++){
(function(n){
divs[i].onclick=function(){
alert(n);
}
})(i)
}
}

2、内部函数

因为是刚开始学习js,所以就从基础知识说起,什么是内部函数呢?说白了,内部函数就是定义在另一个函数中的函数,例如:

function Fn(){
function subFn(){
};//subFn就是一个内部函数,它定义在Fn函数里面。
}

由于subFn被包含在Fn里,这就意味着在Fn内部调用subFn函数是有效的,而在Fn外部调用subFn则是无效的。下面的代码会导致一个javascript的错误:

function Fn(){
document.write("Fn");
function subFn(){
document.write("subFn");
}
}
subFn();

不过在Fn内部调用subFn函数则是有效地:

function Fn(){
document.write("Fn");
function subFn(){
document.write("subFn");
}
subFn();
}
Fn();

2.1、伟大的逃脱

既然目前这样我们不能够在Fn函数外部调用subFn,那怎么才能实现让Fn函数外部也能调用subFn呢?javascript运行开发人员像传递任何类型的数据一样传递函数(这是关键),也就是说,javascript中的内部函数可以逃脱定义他们的外部函数。

(1)逃脱的方法有很多中:比如可以将一个内部函数指定给一个全局变量:

function Fn(){
document.write("Fn");
function subFn(){
document.write("subFn");
}
global=subFn;
}
Fn();
global();

当在调用Fn()函数时,会修改全局变量global,这时候它的引用为subFn;此后调用global和subFn是一样的效果。如果在Fn外部直接调用subFn仍然是会出错的,因为内部函数虽然通过引用保存在全局变量中实现了逃脱,但这个函数的名字依然只存在于Fn的作用域中。

(2)也可以通过父函数的返回值来获得内部函数的引用。

function Fn(){
document.write("Fn");
function subFn(){
document.write("subFn");
}
return subFn;
}
var subRef=Fn();
subRef();

这里并没有在Fn的内部修改全局变量,而是返回了一个对subFn的引用,通过调用Fn这个函数可以获取subFn的引用,而且这个引用可以保存在变量中。

这种即使离开函数作用域的情况下仍能够调用内部函数的事实,意味着只要存在调用内部函数的可能,javascript就要保留被引用的函数。而且javascript运行时需要跟踪引用这个内部函数的所有变量,直到最后一个变量废弃,javascript垃圾收集器才能释放相应的内存空间。

说了半天终于说道和闭包的关系了,闭包是指有权访问另一个函数作用域的变量的函数,创建闭包的常见方式则是在一个函数内部创建另一个函数,就是我们上述说的内部函数。所以刚才说的不是废话,也是和闭包有关的。

1.2、变量的作用域

内部函数也可以有自己的变量,这些变量都被限制在内部函数的作用域中。

function Fn(){
document.write("Fn");
function subFn(){
var subVar=0;
subVar++;
document.write("subFn\t");
document.write("subVar="+subVar+"<br/>");
}
return subFn;
}
var subRef=Fn();
subRef();
subRef();
var subRef2=Fn();
subRef2();
subRef2();

每当通过引用和其他方式调用这个内部函数时,就会创建一个新的subVar变量,然后加1,得出的结果如下:

Fn
subFn innerVar = 1
subFn innerVar = 1
Fn
subFn innerVar = 1
subFn innerVar = 1

内部函数也可以像其他函数一样引用全局变量。

var global=0;
function Fn(){
document.write("Fn");
function subFn(){
global++;
document.write("subFn\t");
document.write("globalVar="+global+"<br/>");
}
return subFn();
}
var subRef=Fn();
subRef();
subRef();
var subRef2=Fn();
subRef2();
subRef2();

现在每次调用这个内部函数都会持续的递增这个全局变量的值。

Fn
subFn globalVar = 1
subFn globalVar = 2
Fn
subFn globalVar = 3
subFn globalVar = 4

如果这个变量是父函数的局部变量呢?因为内部函数会引用到父函数的作用域,内部函数也可以引用到这些变量。

function Fn() {
var outerVar = 0;
document.write("Fn<br/>");
function subFn() {
outerVar++;
document.write("subFn\t");
document.write("outerVar = " + outerVar + "<br/>");
}
return subFn;
}
var subRef = Fn();
subRef ();
subRef ();
var subRef2 = outerFn();
subRef2();
subRef2();

这次的结果非常有意思,也许会出乎我们的意料。

Fn
subFn outerVar = 1
subFn outerVar = 2
Fn
subFn outerVar = 1
subFn outerVar = 2

我们看到的是前两种情况合成的效果,通过每个引用调用的subFn都会独立递增的outerVar,也就是说第二次调用Fn并没有继续沿用outerVar的值,而是在第二次调用的作用域创建并绑定了一个新的outerVar实例,两个计数器无关。

当内部函数在定义他的作用域外部被引用时,就创建了该内部函数的一个闭包,这种情况下我们称既不是内部函数局部变量,也不是其参数的变量为自由变量,称外部函数的调用坏境为封闭闭包的坏境,从本质上讲,如果内部函数引用了外部函数的变量,相当于授权了该变量能够被延迟使用,因此当外部函数被调用完后,这些变量的内存不会被释放(最后的值会被保存),闭包仍然需要使用它们。

3、解惑

现在回过头来看刚才那个经典错误。就会明白为什么每次都弹出divs.length;

function A(){
var divs=document.getElementsByTagName("div");
for(var i=0; i<divs.length; i++){
divs[i].onclick=function(){
alert(i);
}
}
}

上面的代码在页面加载后就会执行,当i的值为divs.length时判断条件不成立,for循环执行完毕,但是因为每个divs的每个onclick方法为内部函数,所以i被闭包引用,内存不能销毁,i的值一直保持divs.length,直到程序改变它或者所有的onclick函数被销毁(主动把函数设为null或者页面卸载)时才会被回收,这样我每点击一次div,onclick函数就会查找i的值,一查等于divs.length,所以就每次都弹出的是divs .length;而第二种方法是使用了一个立即执行的函数又创建了一个闭包,函数声明方法括号内就成了表达式,后面在加上括号,括号就是调用咯,这是把i当参数传入,函数立即执行,num保存每次i的值。

这样,大家对闭包有一定的了解了吧。当然完全了解的话需要把函数的执行坏境和作用域链搞清楚。我个人觉得还蛮详细的。辛苦了~~~

深入理解javascript中的闭包!(转)的更多相关文章

  1. 深入理解JavaScript中的闭包

    闭包没有想象的那么简单 闭包的概念在JavaScript中占据了十分重要的地位,有不少开发者分不清匿名函数和闭包的概念,把它们混为一谈,我希望借这篇文章能够让大家对闭包有一个清晰的认识. 大家都知道变 ...

  2. 【原】理解javascript中的闭包

    闭包在javascript来说是比较重要的概念,平时工作中也是用的比较多的一项技术.下来对其进行一个小小的总结 什么是闭包? 官方说法: 闭包是指有权访问另一个函数作用域中的变量的函数.创建闭包的常见 ...

  3. 【原】理解javascript中的闭包(***********************************************)

    阅读目录 什么是闭包? 闭包的特性 闭包的作用: 闭包的代码示例 注意事项 总结 闭包在javascript来说是比较重要的概念,平时工作中也是用的比较多的一项技术.下来对其进行一个小小的总结 回到顶 ...

  4. 全面理解JavaScript中的闭包的含义及用法

    1.什么是闭包 闭包:闭包就是能够读取其他函数内部变量的函数;闭包简单理解成“定义在一个函数内部的函数”. 闭包的形式:即内部函数能够使用它所在级别的外部函数的参数,属性或者内部函数等,并且能在包含它 ...

  5. 理解JavaScript中的闭包

    (这篇文章后面关于onclick事件的解释是错误的,请不要被误导了2016.6.16) 闭包这个概念给JavaScript初学者心中留下了巨大的阴影,网络上关于闭包的文章不可谓不多,但是能让初学者看懂 ...

  6. 理解 JavaScript 中的 this

    前言 理解this是我们要深入理解 JavaScript 中必不可少的一个步骤,同时只有理解了 this,你才能更加清晰地写出与自己预期一致的 JavaScript 代码. 本文是这系列的第三篇,往期 ...

  7. [译]Javascript中的闭包(closures)

    本文翻译youtube上的up主kudvenkat的javascript tutorial播放单 源地址在此: https://www.youtube.com/watch?v=PMsVM7rjupU& ...

  8. JavaScript中的闭包理解

    原创文章,转载请注明:JavaScript中的闭包理解  By Lucio.Yang 1.JavaScript闭包 在小学期开发项目的时候,用node.js开发了服务器,过程中遇到了node.js的第 ...

  9. 深入理解javascript原型和闭包 (转)

    该教程绕开了javascript的一些基本的语法知识,直接讲解javascript中最难理解的两个部分,也是和其他主流面向对象语言区别最大的两个部分--原型和闭包,当然,肯定少不了原型链和作用域链.帮 ...

随机推荐

  1. iOS开发笔记系列-基础4(变量与数据类型)

    对象的初始化 对象的初始化方法一般都如下: -(id)init { self=[super init]; if(self){ ... } return self; } 这个方法首先会调用父类的初始化方 ...

  2. 给定表达式[x/2] + y + x * y, 其中x,y都是正整数。

    改进了一下,不过还是要十多秒吧. package com.boco.study; import java.math.BigDecimal; import java.util.Calendar; imp ...

  3. How to Tune Java Garbage Collection--reference

    reference:http://architects.dzone.com/articles/how-tune-java-garbage The Performance Zone is support ...

  4. 【排障】编译安装Mysql并使用自启动脚本mysqld后报错

    本文用于记录在某次个人实验搭建DZ论坛,在编译安装部署mysql环节时出的错到最终排除错误的过程, 前面采用DZ官网所采用的编译安装mysql的过程就省去,主要从报错处开始讲述. (题外话,经此一役后 ...

  5. java基础学习总结一(java语言发展历史、jdk的下载安装以及配置环境变量)

    最近一段时间计划复习一下java基础知识,使用的视频课程是尚学堂高淇老师的,上课过程中的心得体会直接总结一下,方便以后复习. 一:计算机语言的发展 1:机器语言,最原始的语言,主要有“01”构成,最早 ...

  6. PV模型

    你想建设一个能承受500万PV/每天的网站吗? 500万PV是什么概念?服务器每秒要处理多少个请求才能应对?如果计算呢? 一.PV是什么 PV是page view的简写.PV是指页面的访问次数,每打开 ...

  7. javascript中substring和substr方法

    1.substring 方法 定义:用于提取字符串中介于两个指定下标之间的字符 语法:stringObject.substring(start,stop) 参数描述: start 必需.一个非负的整数 ...

  8. 2 WPF之XMAL----XMAL概览

    转载:http://blog.csdn.net/fwj380891124/article/details/8085458 微软为了把开发模式从网络开发移植到桌面开发和富媒体网络程序的开发上,微软创造了 ...

  9. 20160510--hibernate懒加载问题

    懒加载 通过asm和cglib二个包实现:Domain是非final的. 1.session.load懒加载. 2.one-to-one(元素)懒加载: 必需同时满足下面三个条件时才能实现懒加载 (主 ...

  10. let 与 expr Shell运算比较 let强强胜出

    Shell脚本中 整数运算一般通过 let 和 expr 这两个指令来实现,如对变量 s 加 1 可以写作:let "s = $s + 1" 或者 s=`expr $s + 1'两 ...