闭包

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. php多图上传问题笔记

    图片上传好用插件有,比如 uploadify  ueditor html5的各种ajax上传插件,大部分都是异步,返回只是true之类,有些时候需要上传图片需要一起上传,其实可以通过操作流程来避免这个 ...

  2. elasticsearch运维实战之2 - 系统性能调优

    elasticsearch性能调优 集群规划 独立的master节点,不存储数据, 数量不少于2 数据节点(Data Node) 查询节点(Query Node),起到负载均衡的作用 Linux系统参 ...

  3. Request 、Response 与Server的使用

    纯属记录总结,以下图片都是来自 ASP.NET笔记之 Request .Response 与Server的使用 Request Response Server 关于Server.MapPath 方法看 ...

  4. 过滤关键字防止XSS攻击

    public static string ClearXSS(string str) { string returnValue = str; if (string.IsNullOrEmpty(retur ...

  5. How to change hostname on SLE

    修改/etc/HOSTNAME文件,在此文件中保存主机名,例如: linuxserv1 然后运行命令设置主机名 # /etc/rc.d/boot.localnet start 方法3. 运行 sysc ...

  6. 群晖SVN Server远程访问

    打开路由器访问界面 选择转发规则->端口映射-新建 在弹出的界面中填写相应的端口号了内网ip 填写svn所在地址的IP,比如:192.168.30.2 添加映射端口,比如svn的默认端口是330 ...

  7. 动画黄金搭档:CADisplayLink&CAShapeLayer

    我们在开发中有时会遇到一些看似非常复杂的动画,不知该如何下手,今天的这篇文章中我会讲到如何利用CADisplayLink和CAShapeLayer来构建一些复杂的动画,希望能在你下次构建动画中,给你一 ...

  8. Redmine新建问题速度慢

    Redmine有时候新建问题 ,更新指派人的时候反应很慢, 很大原因应该是发送邮件方式不对. 1.一种方式是改为异步发送      2.另外检测到Redmine日志 ,会发现发送邮件失败 ,也会导致发 ...

  9. [转]ORACLE函数大全

    SQL中的单记录函数 1.ASCII返回与指定的字符对应的十进制数;SQL> select ascii('A') A,ascii('a') a,ascii('0') zero,ascii(' ' ...

  10. Android Studio中获取查看签名SHA1证书指纹数据或MD5的方法

    原来在Eclipse中获取SHA1或者MD5,在IDE界面上就可以查找到. 切换到Android Studio后,如何查看呢?找了半天没找到.那就老办法命令行. 第一步.打开Android Studi ...