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

本文深入探讨了JavaScript中的闭包概念,详细解释了闭包域的形成与作用,并通过一个经典的面试题目来阐述闭包可能导致的问题及其多种解决方案。

原文地址:https://segmentfault.com/a/1190000003818163

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

  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>

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

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

<think>我们正在处理关于JavaScript闭包面试题的用户查询。根据提供的引用,特别是引用[2]、[3]、[4],我们可以总结出几个常见的闭包面试题及其解析。用户要求的是“JavaScript闭包面试题”,因此我们将重点放在常见的闭包面试题上,并给出解析。根据引用[3]和[4]中的示例,有一个非常典型的闭包面试题,涉及一个函数返回一个对象,对象中有一个方法,该方法又调用外部函数,形成闭包。我们将以这个题目为例进行解析。题目如下(来自引用[4]):```javascriptfunctionfun(n,o){console.log(o);return{fun:function(m){returnfun(m,n);}};}```然后有如下调用:```javascriptvara=fun(0);//?a.fun(1);//?a.fun(2);//?a.fun(3);//?varb=fun(0).fun(1).fun(2).fun(3);//?varc=fun(0).fun(1);//?c.fun(2);//?c.fun(3);//?```我们将逐步分析这个题目。首先,函数`fun`接受两个参数`n`和`o`,并打印出`o`。然后返回一个对象,该对象有一个方法`fun`,这个方法接受一个参数`m`,并调用`fun(m,n)`(注意这里调用的是外部的`fun`函数,并且将当前的`n`作为第二个参数传入,即新的调用中`o`将是当前的`n`)。分析关键点:1.每次调用外部函数`fun`时,都会创建一个新的闭包,该闭包保存了此次调用时的参数`n`(因为内部函数引用了`n`)。2.内部函数`fun`(即返回对象的方法)被调用时,会使用闭包中保存的`n`作为下一次调用的第二个参数(即`o`)。现在逐行分析:第一部分:```javascriptvara=fun(0);//调用fun(0),此时n=0,o未传入(即undefined),所以打印undefined//返回一个对象,这个对象的fun方法被调用时,会调用fun(m,0)(因为这里的n是0)//将返回的对象赋给aa.fun(1);//调用a.fun(1)->即调用fun(1,0)(因为闭包中保存的n是0)//此时,在fun(1,0)中:n=1,o=0->打印0//然后返回一个新对象,这个新对象的fun方法被调用时,会调用fun(m,1)(因为这里的n是1)a.fun(2);//同样,调用a.fun(2)->调用fun(2,0)(注意:a是第一次调用返回的对象,其内部闭包保存的n是0)//所以打印0a.fun(3);//同理,打印0```所以第一部分的输出是:`undefined,0,0,0`第二部分:```javascriptvarb=fun(0).fun(1).fun(2).fun(3);//分解://1.fun(0):打印undefined,返回对象A(对象A的fun方法调用时会用n=0)//2.对象A.fun(1):调用fun(1,0)->打印0,返回对象B(对象B的fun方法调用时会用n=1)//3.对象B.fun(2):调用fun(2,1)->打印1,返回对象C(对象C的fun方法调用时会用n=2)//4.对象C.fun(3):调用fun(3,2)->打印2//所以输出:undefined,0,1,2```第三部分:```javascriptvarc=fun(0).fun(1);//1.fun(0):打印undefined,返回对象A//2.对象A.fun(1):调用fun(1,0)->打印0,返回对象B(此时对象B的闭包中n=1)//将对象B赋给cc.fun(2);//调用c(即对象B)的fun(2)->调用fun(2,1)->打印1c.fun(3);//调用c的fun(3)->调用fun(3,1)->打印1```所以第三部分的输出:`undefined,0`(来自`varc=...`),然后`1,1`(来自`c.fun(2)`和`c.fun(3)`)。因此,整个题目的输出为:```undefined000------------undefined012------------undefined011```这个题目考察了闭包的概念:内部函数可以访问外部函数的变量,并且每次调用外部函数都会生成一个新的闭包环境。除了这个题目,闭包面试题还可能涉及循环中的闭包闭包的作用域链、闭包的内存管理等。例如,循环中使用闭包经典问题(来自引用[1]和[2]):```javascriptfor(vari=0;i<5;i++){setTimeout(function(){console.log(i);},1000);}```这段代码会输出5个5,因为setTimeout中的函数共享同一个闭包,而变量i是全局作用域(或函数作用域)中的,循环结束后i变成了5。解决方法通常是使用立即执行函数表达式(IIFE)或使用let声明块级作用域变量。根据用户的需求,我们提供以上两个典型的闭包面试题及其解析。现在,我们按照要求生成相关问题。</think>### JavaScript 闭包面试题解析 #### 典型闭包面试题分析(引用[3][4]) ```javascript function fun(n, o) { console.log(o); return { fn: function(m) { return fun(m, n); } }; } // 示例调用 var a = fun(0); // 输出:undefined a.fn(1); // 输出:0 a.fn(2); // 输出:0 a.fn(3); // 输出:0 var b = fun(0).fn(1).fn(2).fn(3); // 输出:undefined, 0, 1, 2 var c = fun(0).fn(1); // 输出:undefined, 0 c.fn(2); // 输出:1 c.fn(3); // 输出:1 ``` #### 闭包机制解析 1. **闭包形成原理** 函数`fun`返回的对象中包含内部函数`fn`,该函数引用了外部变量`n`,形成闭包。每次调用`fun`
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值