美文网首页
2020-10-22随笔 闭包及经典题目的追问

2020-10-22随笔 闭包及经典题目的追问

作者: JLong | 来源:发表于2020-10-22 09:11 被阅读0次
    闭包经典题

    这段代码很短,只有 7 行,我想,能读到这里的同学应该不需要我逐行解释这段代码在做什么吧。候选人面对这段代码时给出的结果也不尽相同,以下是典型的答案:

    A. 20% 的人会快速扫描代码,然后给出结果:0,1,2,3,4,5;

    B. 30% 的人会拿着代码逐行看,然后给出结果:5,0,1,2,3,4;

    C. 50% 的人会拿着代码仔细琢磨,然后给出结果:5,5,5,5,5,5;

    只要你对 JS 中同步和异步代码的区别、变量作用域、闭包等概念有正确的理解,就知道正确答案是 C,代码的实际输出是:

    2017-03-18T00:43:45.873Z 5

    2017-03-18T00:43:46.866Z 5

    2017-03-18T00:43:46.868Z 5

    2017-03-18T00:43:46.868Z 5

    2017-03-18T00:43:46.868Z 5

    2017-03-18T00:43:46.868Z 5

    接下来我会追问:如果我们约定,用箭头表示其前后的两次输出之间有 1 秒的时间间隔,而逗号表示其前后的两次输出之间的时间间隔可以忽略,代码实际运行的结果该如何描述?会有下面两种答案:

    A. 60% 的人会描述为:5 -> 5 -> 5 -> 5 -> 5,即每个 5 之间都有 1 秒的时间间隔;

    B. 40% 的人会描述为:5 -> 5,5,5,5,5,即第 1 个 5 直接输出,1 秒之后,输出 5 个 5;

    这就要求候选人对 JS 中的定时器工作机制非常熟悉,循环执行过程中,几乎同时设置了 5 个定时器,一般情况下,这些定时器都会在 1 秒之后触发,而循环完的输出是立即执行的,显而易见,正确的描述是 B。

    追问 1:闭包

    如果这道题仅仅是考察候选人对 JS 异步代码、变量作用域的理解,局限性未免太大,接下来我会追问,如果期望代码的输出变成:5 -> 0,1,2,3,4,该怎么改造代码?熟悉闭包的同学很快能给出下面的解决办法:

    for (var i = 0; i < 5; i++) {

        (function(j) {  // j = i

            setTimeout(function() {

                console.log(new Date, j);

            }, 1000);

        })(i);

    }

    console.log(new Date, i);

    巧妙的利用IIFE(Immediately Invoked Function Expression:声明即执行的函数表达式)来解决闭包造成的问题,确实是不错的思路,但是初学者可能并不觉得这样的代码很好懂,至少笔者初入门的时候这里琢磨了一会儿才真正理解。

    增补:如果有同学给出如下的解决方案,则说明他是一个仔细看API 文档的人,这种习惯会让他学习的时候少走弯路,具体代码如下:

    var output = function (i) {

        setTimeout(function() {

            console.log(new Date, i);

        }, 1000);

    };

    for (var i = 0; i < 5; i++) {

        output(i);  // 这里传过去的 i 值被复制了

    }

    console.log(new Date, i);

    有没有更符合直觉的做法?答案是有,我们只需要对循环体稍做手脚,让负责输出的那段代码能拿到每次循环的i值即可。该怎么做呢?利用 JS 中基本类型(Primitive Type)的参数传递是按值传递(Pass by Value)的特征,不难改造出下面的代码:

    var output = function (i) {    setTimeout(function() {        console.log(new Date, i);    }, 1000);};for (var i = 0; i < 5; i++) {    output(i);  // 这里传过去的 i 值被复制了}console.log(new Date, i);复制代码

    能给出上述 2 种解决方案的候选人可以认为对 JS 基础的理解和运用是不错的,可以各加 10 分。当然实际面试中还有候选人给出如下的代码:

    for (let i = 0; i < 5; i++) {

        setTimeout(function() {

            console.log(new Date, i);

        }, 1000);

    }

    console.log(new Date, i);

    细心的同学会发现,这里只有个非常细微的变动,即使用 ES6块级作用域(Block Scope)中的let替代了var,但是代码在实际运行时会报错,因为最后那个输出使用的i在其所在的作用域中并不存在,i只存在于循环内部。

    摘抄自https://juejin.im/post/6844903474212143117#heading-0,更详细可以自行浏览

    相关文章

      网友评论

          本文标题:2020-10-22随笔 闭包及经典题目的追问

          本文链接:https://www.haomeiwen.com/subject/nvcumktx.html