美文网首页
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随笔 闭包及经典题目的追问

    这段代码很短,只有 7 行,我想,能读到这里的同学应该不需要我逐行解释这段代码在做什么吧。候选人面对这段代码时给出...

  • 闭包经典题

    小测试 手动想想为何输出以上值

  • 闭包经典题

    使用var或是非对象内部的函数表达式内,可以访问到存放当前函数的变量;在对象内部的不能访问到。 原因也非常简单,因...

  • 闭包经典题

    click me click me click me click me var elements = d...

  • 闭包经典题(答案及解释)

  • JavaScript深入之闭包

    JavaScript深入系列第八篇,介绍理论上的闭包和实践上的闭包,以及从作用域链的角度解析经典的闭包题。 定义 ...

  • 闭包(closure)

    ● 闭包基础 ● 闭包作用 ● 闭包经典例子 ● 闭包应用 ● 闭包缺点 ● 参考资料 1、闭包基础 作用域和作...

  • Python闭包

    闭包 = 环境变量 + 函数 调用闭包内部的环境变量 闭包的经典误区 闭包与非闭包实现人类走路 非闭包 闭包

  • iOS开发之Swift篇(7)—— 闭包

    目录 版本 闭包的定义及使用 尾随闭包 逃逸闭包 版本 Xcode 11.0Swift 5.1 闭包的定义及使用 ...

  • 闭包的理解

    闭包目的: 闭包三个特性: 闭包的用途: 闭包的优点: 闭包的缺点: 参考文献: https://zhuanlan...

网友评论

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

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