关于setTimeout的面试题

2018-07-13  本文已影响0人  mills_han

经常看到网上的前端面试题中会有关于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秒之后才能执行

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

上一篇 下一篇

猜你喜欢

热点阅读