转载自:http://mp.weixin.qq.com/s/-ahvLmHaiIpUah-24eMTfA

闭包:

1.正确的说,应该是指一个闭包域,每当声明了一个函数,它就产生了一个闭包域(可以解释为每个函数都有自己的函数栈),每个闭包域(Function 对象)都有一个 function scope(不是属性),function scope内默认有个名为Global的全局引用(有了这个引用,就可以直接调用 Global 的属性或方法)

2.凡是在闭包域内声明的变量或方法,外部无法直接访问

3.闭包域可以访问外部的变量或方法 
(上图为 chrome 下 debug 环境)

当在一个闭包域内包含另一个闭包域时(简单的说就是在一个函数内有另一个函数,当然这个内部函数的生命周期是依附于外部函数的), 此时,若子闭包域(内部的闭包域,内部函数)使用了父闭包域(外部闭包域,外部函数)的私有变量(在父闭包域中声明的变量,父闭包域的外部空间无法直接访问,但子闭包域可以访问),子闭包域即当前的子函数的 function scope 会产生一个 closure 对象属性,这个对象属性内包含的是子闭包域对父闭包域的所有引用(只要子闭包域(内部函数)还存活,其父闭包域(外部函数)就依旧存活),倘若在父闭包域存活期间对其私有变量内容进行修改,则对这些父闭包域的私有变量进行引用的子闭包域中 function scope 的 closure 对象属性的内容也会发生变化,因为这只是引用.

举例:

<!DOCTYPE html><html lang="en"><head><meta charset="UTF-8"><title></title></head><body><script type="text/javascript" charset="utf-8">//函数 a 有一个私有变量 p 和一个内部函数 innerAfunction a() {                      //外部闭包域 ,一个名为 a 的 Function 对象var p = 0;                      //私有变量 pvar innerA = function () {      //内部闭包域 ,一个名为 innerA 的 Function 对象
console.log(p);             //对外部闭包域的私有变量进行了引用,故 innerA 对象的 function scope 会产生一个名为 closure 的对象属性,closure 对象内含有一个名为 p 的引用
}innerA();//输出0
p++;innerA();//输出1
}a();</script></body></html>

结果如下:

第一次调用innerA

第二次调用 innerA

控制台输出

回到主题 面试经典问题

<!DOCTYPE html><html lang="en"><head><meta charset="UTF-8"><title></title><script type="text/javascript">//面试经典问题:function onMyLoad(){/*抛出问题:此题的目的是想每次点击对应目标时弹出对应的数字下标 0~4,但实际是无论点击哪个目标都会弹出数字5问题所在:arr 中的每一项的 onclick 均为一个函数实例(Function 对象),这个函数实例也产生了一个闭包域,这个闭包域引用了外部闭包域的变量,其 function scope 的 closure 对象有个名为 i 的引用,外部闭包域的私有变量内容发生变化,内部闭包域得到的值自然会发生改变*/var arr = document.getElementsByTagName("p");for(var i = 0; i < arr.length;i++){arr[i].onclick = function(){alert(i);}}}</script></head><body οnlοad="onMyLoad()"><p>产品一</p><p>产品二</p><p>产品三</p><p>产品四</p><p>产品五</p></body></html>

解决办法:

解决办法一

/*解决思路:增加若干个对应的闭包域空间(这里采用的是匿名函数),专门用来存储原先需要引用的内容(下标),不过只限于基本类型(基本类型值传递,对象类型引用传递)*/for(var i = 0;i<arr.length;i++){//声明一个匿名函数,若传进来的是基本类型则为值传递,故不会对实参产生影响,//该函数对象有一个本地私有变量arg(形参) ,该函数的 function scope 的 closure 对象属性有两个引用,一个是 arr,一个是 i//尽管引用 i 的值随外部改变 ,但本地私有变量(形参) arg 不会受影响,其值在一开始被调用的时候就决定了.
(function (arg) {arr[i].onclick = function () {  //onclick函数实例的 function scope 的 closure 对象属性有一个引用 arg,
alert(arg);                 //只要 外部空间的 arg 不变,这里的引用值当然不会改变
}})(i);                              //立刻执行该匿名函数,传递下标 i(实参)

}

解决办法二

/*解决思路:将下标作为对象属性(name:"i",value:i的值)添加到每个数组项(p对象)中*/for(var i = 0;i<arr.length;i++){//为当前数组项即当前 p 对象添加一个名为 i 的属性,值为循环体的 i 变量的值,//此时当前 p 对象的 i 属性并不是对循环体的 i 变量的引用,而是一个独立p 对象的属性,属性值在声明的时候就确定了//(基本类型的值都是存在栈中的,当有一个基本类型变量声明其等于另一个基本变量时,此时并不是两个基本类型变量都指向一个值,而是各自有各自的值,但值是相等的)
arr[i].i = i;arr[i].onclick = function () {alert(this.i);}}

解决办法三

/*解决思路:与解决办法一有点相似但却有点不太相似.相似点:同样是增加若干个对应的闭包域空间用来存储下标不同点:解决办法一是在新增的匿名闭包空间内完成事件的绑定,而此例是将事件绑定在新增的匿名函数返回的函数上此时绑定的函数中的 function scope 中的 closure 对象的 引用 arg 是指向将其返回的匿名函数的私有变量 arg*/for(var i = 0; i<arr.length;i++){arr[i].onclick = (function(arg){return function () {alert(arg);}})(i);}

解决办法四

/*解决思路与解决办法一相同*/for(var i = 0; i<arr.length;i++){(function(){var temp = i;arr[i].onclick = function () {alert(temp);}})();}

解决办法五

/*解决思路与解决办法三及四相同*/for(var i = 0;i<arr.length;i++){arr[i].onclick = (function () {var temp = i;return function () {alert(temp);}})();}

解决办法六

/*解决思路:将下标添加为绑定函数的属性*/for(var i = 0;i<arr.length;i++){(arr[i].onclick = function () {alert(arguments.callee.i);      //arguments 参数对象  arguments.callee 参数对象所属函数
}).i = i;}

解决办法七

/*解决思路:通过 new 使用 Function 的构造函数 创建 Function 实例实现,由于传入的函数体的内容是字符串,故 Function 得到的是一个字符串拷贝,而没有得到 i 的引用(这里是先获取 i.toString()然后与前后字符串拼接成一个新的字符串,Function 对其进行反向解析成 JS 代码)*/for(var i = 0;i<arr.length;i++){arr[i].onclick = new Function("alert("+i+");");//每 new 一个 Function 得到一个 Function 对象(一个函数),有自己的闭包域

}

解决办法八

/*解决思路:直接通过 Function 返回一个函数与解决办法七的不同之处在于:解决办法七使用 new,使用了 new,此时 Function 函数就被当成构造器可以用来构造一个 Function 实例返回当前解决办法没有使用 new ,即将 Function 函数当成一个函数,传入参数返回一个新函数;其实此处 new 与不 new 只是的区别在于:使用了 new 即 Function 函数充当构造器,由 JS 解析器生产一个新的对象,构造器内的 this 指向该新对象;不实用 new 即 Function 函数依旧是函数,由函数内部自己生产一个实例返回.*/for(var i = 0;i<arr.length;i++){arr[i].onclick = Function("alert("+i+");");}

解决办法九

使用ES6新语法 let 关键字 由于几新东西 各浏览器支持不同
chrome 及 opera支持以下语法<script type="application/javascript">"use strict";//使用严格模式,否则报错 SyntaxError: Block-scoped declarations (let, const, function, class) not yet supported outside strict modevar arr = document.getElementsByTagName("p");for(var i = 0;i<arr.length;i++){let j = i;//创建一个块级变量
arr[i].onclick = function () {alert(j);}}</script>

在 chrome 查看

可以在控制台看到 j 变量是一个 block 级的变量

待函数绑定完成后看数组项:

此时的该数组项的<function scope>的 Block 域有个 j 存储的就是对应的数组下标 
firefox支持一下语法

<script type="application/javascript;version=1.7">var arr = document.getElementsByTagName("p");for(var i = 0;i<arr.length;i++){let j = i;arr[i].onclick = function () {alert(j);}}</script>

由于新语法各大厂商的支持尚未规范故暂不不推荐使用

解决办法大同小异,只要理解其中的实质,可以写出多多的解决办法

转载于:https://www.cnblogs.com/guojikun/articles/6369842.html

用9种办法解决 for 循环取 i相关推荐

  1. 苹果手机上滑动会卡顿_7种办法解决苹果手机卡顿 让你的手机用起来如丝般顺滑...

    原标题:7种办法解决苹果手机卡顿 让你的手机用起来如丝般顺滑 很多人都有这种体验,刚买的手机用起来特别爽,不管点哪个APP都是秒开,随着时间的推移,越来越卡顿,甚至有的时候直接卡死,无奈之下只好重新开 ...

  2. 小米手机html无法,小米手机无法开机进入不了桌面怎么解决 两种办法解决小米手机无法开机问题...

    小米手机无法开机进入不了桌面怎么办,出现无法开机我们一个是可以刷机来解决,另一个临时办法就清除数据恢复出厂设置试一下,具体的两种解决无法开机问题,下面小编都为各位介绍 注意:不管是什么原因导致你手机无 ...

  3. 小米5进系统无显示无服务器,小米手机无法开机进入不了桌面怎么解决 两种办法解决小米手机无法开机问题...

    注意:不管是什么原因导致你手机无法开机我们都可以参考下面的方法来解决了,具体的如下. 方法一:小米手机无法进入系统的常用方法 1 既然不能开机了那肯定是已经关机了,我们现在只要同时按下开机键和音量放大 ...

  4. 用9种办法解决 JS 闭包经典面试题之 for 循环取 i

    2017-01-06 Tomson JavaScript 转自 https://segmentfault.com/a/1190000003818163 闭包 1.正确的说,应该是指一个闭包域,每当声明 ...

  5. java 多线程跑数据_java——多线程的实现方式、三种办法解决线程赛跑、多线程数据同步(synchronized)、死锁...

    多线程的实现方式:demo1.demo2 demo1:继承Thread类,重写run()方法 packagethread_test;public class ThreadDemo1 extendsTh ...

  6. 苹果机内存满开不了机怎么办?三种办法解决这个问题。

    iPhone手机使用比较顺畅,但难免也会遇到因为内存不足而导致无法正常开机的情况,很多小伙伴一开始都不相信iPhone内存不足会开不了机,可是如果不幸遇到了这个问题,又该如何解决呢?接下来,就给大家分 ...

  7. html中怎么在新窗口打开文件夹,如何解决文件夹总是在新窗口打开 三种办法解决文件夹总是在新窗口打开...

    之前用电脑打开文件夹时,都是在同一窗口,很方便,但是,不知何故,每次打开文件夹时,都会创建一个新窗口,如果文件路径很短,还只有二三个窗口,如果路径很长,要打开七八个窗口,甚至更多,而且后面还要一个一个 ...

  8. 计算机窗口总是新窗口打开,如何解决文件夹总是在新窗口打开 三种办法解决文件夹总是在新窗口打开...

    之前用电脑打开文件夹时,都是在同一窗口,很方便,但是,不知何故,每次打开文件夹时,都会创建一个新窗口,如果文件路径很短,还只有二三个窗口,如果路径很长,要打开七八个窗口,甚至更多,而且后面还要一个一个 ...

  9. iphone死机屏幕没反应?可以用这2种办法解决!

    iPhone用的时间长了,难免不会遇到卡屏.死机的情况,如果出现这种状况我们应该怎么办呢,下面小编整理出来了几招解决方法,教大家解决iPhone卡屏.死机的问题. 一.强制重启 如果自己的iPhone ...

最新文章

  1. 基于HTML5的Google水下搜索
  2. LeetCode--258--各位相加*
  3. python里面的类和对象_Python中类和对象在内存中是如何保存?
  4. Go 语言编程 — go 常用指令
  5. dhcp 中继代理配置方法
  6. dbunit java_Java – 让DbUnit使用Hibernate事务
  7. ORA-00972: identifier is too long问题解决
  8. STM32F407+CubeMX - 使用GPIO翻转+示波器测量函数的执行时间
  9. java -jar 怎么停止_图解Java日志体系
  10. Windows学习总结(7)——学会CMD命令提示符的重要性
  11. php imagedashedline,如何用php作线形图的函数
  12. mysql datapump_mysqlpump - 一个数据库备份程序
  13. Ros学习笔记(六)Launch启动文件
  14. 提高MyEclipse启动速度
  15. 安装R软件(R、studio)的安装包下载官网和教程
  16. android模拟器多点触摸,Windows 7 多点触摸开发模拟器和多点触摸驱动
  17. IBATIS开发指南(夏昕)
  18. 天津SEO优化:seo优化后期网站关键词排名下降原因
  19. Godot Shader:读屏着色器
  20. dreamweaver快捷键大全

热门文章

  1. 亲密关系沟通-【唤起亲密】-在平淡关系中创造高质量沟通
  2. idea怎么拉不同的git分支_idea中git分支、合并与使用
  3. 小程序接入h5页面_原生小程序接入H5页面,请求后台接口,获取数据
  4. concurrentbag 删除_你知道吗?这样删除iPhone中的APP腾出的空间会更大
  5. 机器学习实战——KNN及部分函数注解
  6. 光通量发光强度照度亮度关系_照度、强光、光通量之间是什么关系
  7. 华为nova7保密柜_华为nova8系列发布 Vlog视频旗舰3299元起
  8. sql select distinct常见错误_这8种常见的SQL错误用法,80%的程序员还在犯
  9. “技术需求”与“技术成果”项目之间关联度计算模型 复赛top1解决方案
  10. LSTM和GRU的解析从未如此通俗易懂(动图)