用9种方法处理 JS 闭包典范面试题之 for 轮回取 i

闭包

  1. 准确的说,应该是指一个闭包域,每当声清楚明了一个函数,它就发作了一个闭包域(能够解释为每一个函数都有本身的函数栈),每一个闭包域(Function 对象)都有一个 function scope(不是属性),function scope内默许有个名为 Globe 的全局援用(有了这个援用,就能够直接挪用 Globe 的属性或要领)

  2. 通常在闭包域内声明的变量或要领,外部没法直接接见

  3. 闭包域能够接见外部的变量或要领

《用9种方法处理 JS 闭包典范面试题之 for 轮回取 i》
(上图为 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

《用9种方法处理 JS 闭包典范面试题之 for 轮回取 i》

第二次挪用 innerA

《用9种方法处理 JS 闭包典范面试题之 for 轮回取 i》

控制台输出

《用9种方法处理 JS 闭包典范面试题之 for 轮回取 i》

回到主题 口试典范题目

<!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 检察

《用9种方法处理 JS 闭包典范面试题之 for 轮回取 i》

能够在控制台看到 j 变量是一个 block 级的变量

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

《用9种方法处理 JS 闭包典范面试题之 for 轮回取 i》
此时的该数组项的<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>

因为新语法各大厂商的支撑还没有范例故暂不不引荐运用

处理办法迥然差别,只需明白个中的本质,能够写出多多的处理办法

    原文作者:Tomson
    原文地址: https://segmentfault.com/a/1190000003818163
    本文转自网络文章,转载此文章仅为分享知识,如有侵权,请联系博主进行删除。
点赞