关于setTimeout的面试题

经常看到网上的前端面试题中会有关于setTimeout的这道题,这题经常有人写,一道题包含了javascript 的作用域闭包事件循环的知识点,也说明了理解这些知识点对于一个前端开发人员对于JavaScript语言层次的理解程度的重要性。凭着自己的理解也参阅一些资料解答一下这个题,同时也巩固下知识点。
这里先把这道经典的题放出来

//请你预测一下代码会输出什么?
for(var i = 0; i <= 5; i++) {
    setTimeout(function() {
        console.log(i);
    },1000)
}

基本上首先会让你预测下代码的输出
如果你没有理解透js作用域和闭包的知识点的话,你可能会认为这道题的输出顺序是:
for循环,输出1,2,3,4,5
或者循环输出1~5

但是实际的答案在log之后循环输出了五个数字6!!

接着可能面试官会让你改下代码,期望结果是每间隔一秒输出一个数字,即:

等待1秒  输出1,等待2秒 输出2,等到3秒 输出3....

关于知识点

1.作用域

引用《你不知道的javascript》中的一个比喻,可以把作用域链想象成一座高楼,第一层代表当前执行作用域,楼的顶层代表全局作用域。我们在查找变量时会先在当前楼层进行查找,如果没有找到,就会坐电梯前往上一层楼,如果还是没有找到就继续向上找,以此类推。到达顶层后(全局作用域),可能找到了你所需的变量,也可能没找到,但无论如何查找过程都将停止。

2.闭包

我觉得红宝书(JavaScript高级程序设计)中的描述很好理解。
闭包是有权访问另一个函数作用域中的变量的函数。创建闭包的常见方式,就是在一个函数内部创建另一个函数。
例:A函数中定义了B函数并且返回了B函数,那么不管B函数在哪里被调用 如何调用,它都会保留A函数的作用域。

3.事件循环

这个概念会涉及到比较多的概念,篇幅会比较大,下一篇单独抽出来写写。
可以参考阮一峰老师的解说http://www.ruanyifeng.com/blog/2014/10/event-loop.html
这里我们就当知道了事件循环/任务队列的知识点来说。

代码执行顺序分析

回过来看看这段代码的执行顺序,首先for循环执行,在js引擎读到setTimeout时,因为setTimeout不是立即执行的,他们的回调会被push到宏任务队列中,再回头执行任务队列中的回调函数时,变量i早就变成了6。知道了原因,我们着手解决问题。这里我们需要给setTimeout创建一个闭包的环境,让它的回调函数顺利取到循环中的变量i就解决问题了。

这里总结了4种方法,仔细看看这会是很熟悉的写法哦
(1)使用IIFE(立即执行的匿名函数)

//间隔1秒依次输出1,2,3,4, 5
for(var i = 1; i <= 5; i++) {
    (function(i){
        setTimeout(function() {
            console.log(i);
        }, i*1000)
    })(i);
}

(2)使用ES6语法中的let来声明变量i
es6中的let声明的变量是具有块级作用域的,所以我们可以大胆的使用

for(let i = 1;i <=5; i++) {
    setTimeout(function() {
        console.log(i);
    },i*1000)
}

(3)使用bind方法

for(var i = 1; i <= 5; i++) {
     setTimeout(function(i) {
        console.log(i);
    }.bind(null, i),i*1000)
}

(4)利用setTimeout的第三个参数!!

for(var i = 1; i<= 5;i++) {
    setTimeout(function time(i) {
        console.log(i);
    },i*1000,i)
 }

注意:setTineout的第三个参数及以后的参数都可以作为回调函数的参数哦

关于setTimeout的延时参数
setTimeout(function() {
  console.log('代码执行了');
},3000)

我们一般说代码在3秒之后执行,这样的说法是不严谨的。

准确的解释是:3秒后,setTimeout里的函数被推入event queue,而event queue里的任务,只有在主线程空闲下来之后才会去执行。
如果主线程上有很多任务执行,超过3秒,比如执行了10秒,那么这个函数只能在10秒之后才能执行

  • 另外:为了确保浏览器的执行一致,HTML5规范规定设置的最小延迟是4ms

OK,这些就是我所知道的关于setTimeout的全部点了,javascript的知识点零散且庞杂,互勉~

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 第一部分 HTML&CSS整理答案 1. 什么是HTML5? 答:HTML5是最新的HTML标准。 注意:讲述HT...
    kismetajun阅读 27,958评论 1 45
  • 题目如下: 上述代码打印出什么?因为涉及到执行时间,所以结果跟执行的时机有关,下面是我执行的结果: 那么从上面的结...
    小笨郎阅读 668评论 0 1
  • 在线阅读 http://interview.poetries.top[http://interview.poetr...
    前端进阶之旅阅读 115,052评论 24 450
  • 第2章 基本语法 2.1 概述 基本句法和变量 语句 JavaScript程序的执行单位为行(line),也就是一...
    悟名先生阅读 4,272评论 0 13
  • 「 说ino的你一点都不平庸 」 「 这是ino食的第2篇文章 」 很多关于土豆的食谱都很简单,是啊谁还没吃过土豆...
    ino我不阅读 261评论 0 0