闭包

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 和一个内部函数 innerA

       function a() {                      //外部闭包域 ,一个名为 a 的 Function 对象

           var p = 0;                      //私有变量 p

           var 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 onload="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 mode

   var 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>

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

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

170106、用9种办法解决 JS 闭包经典面试题之 for 循环取 i的更多相关文章

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

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

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

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

  3. QThread 爬坑之旅(三种办法解决QObject: Cannot create children for a parent that is in a different thread)

    Cannot create children for a parent that is in a different thread. 在Qt的官方文档,大家知道有两种方式使用QThread. You ...

  4. JS闭包经典例题

    上一篇文章谈论了闭包的概念和一些应用,并给出一个例题,这篇文章就此道例题进行讨论. function fun(n,o) { console.log(o); return { fun:function( ...

  5. JS闭包机制实现为DOM元素循环添加事件

    HTML代码: <button type='button' class='btn' id='1'>按钮1</button> <button type='button' c ...

  6. js 不常用面试题 数组对象深度取值

    function getPersonInfo(one, two, three) { console.log(one); console.log(two); console.log(three); } ...

  7. js高频经典面试题总结

    类型转换问题 console.log(null>=0); console.log(null<=0); console.log(null==0); console.log(undefined ...

  8. js检测数据类型四种办法

    面试题中经常会考js数据类型检测,今天我来分享一下js中常用的四种方法判断数据类型,欢迎指点更正. 废话不多说,直入正题. 1.typeof console.log(typeof "&quo ...

  9. js_html_input中autocomplete="off"在chrom中失效的解决办法 使用JS模拟锚点跳转 js如何获取url参数 C#模拟httpwebrequest请求_向服务器模拟cookie发送 实习期学到的技术(一) LinqPad的变量比较功能 ASP.NET EF 使用LinqPad 快速学习Linq

    js_html_input中autocomplete="off"在chrom中失效的解决办法 分享网上的2种办法: 1-可以在不需要默认填写的input框中设置 autocompl ...

随机推荐

  1. 使用holder进行内存管理

    在C++中,我们使用new 和delete进行自己的内存管理. void test_func() { someType *ptr = new someType; //使用ptr ptr->fun ...

  2. Invoke-Command和-ComputerName 效率比较

    看到网上有文章说Invoke-Command的方式相较其他方式的效率要高,特地试验了一下,但是这个实验不是很好: 机器只有2台 0. 用Get-WinEvent,日志数=200,Invoke方式快 1 ...

  3. 【5集iCore3_ADP演示视频】5-4 iCore3与应用开发平台的组装与拆卸

    iCore3双核心应用开发平台基于iCore3双核心板,包含ARM.FPGA.7寸液晶屏.双通道数字示波器.任意波发生器.电压表等模块,是一款专为电子爱好者设计的综合性电子学习系统. [视频简介]本视 ...

  4. Random随机类(11选5彩票)BigInteger大数据类(华为面试题1000的阶乘)

    先上Java Web图 为了简化叙述,只写Java代码,然后控制台输出 使用[Random类]取得随机数 import java.util.Random; public class Fir { pub ...

  5. ASM FailGroup验证

    ASM-FailGroup验证 一.FailGroup有效性验证 创建DiskGroup,在Redundancy选项 High:至少3块disk,至少3个failgroup,每一个extent存在1主 ...

  6. github搭建静态博客

    p { margin-bottom: 0.1in; line-height: 120% } 1. 创建Repository 创建一个与自己github用户名对应的Repository,例如:abc.g ...

  7. Windows下使用WSRM限制MongoDB内存

    有个项目用到了MongoDB,我们是在WINDOWS 2008 64位环境下部署的,为啥不部署到linux下面呢,我们没那么多服务器,只能将就一下了. 大家都知道Mongodb吃内存太厉害了,如果不重 ...

  8. freebsd 系统时间

    http://blog.csdn.net/wowoto/article/details/5557810 https://www.douban.com/note/150233427/ date #查看当 ...

  9. iOS 保持界面流畅的技巧

    http://blog.ibireme.com/2015/11/12/smooth_user_interfaces_for_ios/

  10. 基于ssh框架开发的购物系统的质量属性

    根据前面的博客,我们已经大致了解了ssh架构开发整体概念:Struts是一个实现了MVC模式的经典的框架:Hibernate是轻量级Java EE应用的持久层解决方案,以面向对象的方式提供了持久化类到 ...