经常看到网上的前端面试题中会有关于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的知识点零散且庞杂,互勉~