首页 > 代码库 > 一道容易栽坑的有趣的面试题(关于js,定时器,闭包等)
一道容易栽坑的有趣的面试题(关于js,定时器,闭包等)
1.首先下面代码输出什么?
1 for (var i = 0; i < 5; i++) {2 console.log(i);3 }
输出:0 1 2 3 4
2.上面只是普通的输出,没有陷阱再看下面这个题(套路开始了)
1 for (var i = 0; i < 5; i++) {2 setTimeout(function() {3 console.log(i);4 }, 1000 * i);5 }
输出:5 5 5 5 5
setTimeout 会延迟执行,那么执行到 console.log 的时候,其实 i 已经变成 5 了,因此会输出5个5
3.上面的题超出我们的预想,我们肯定想得到的是0 1 2 3 4,那么问题来了 怎么才能输出0 1 2 3 4呢
1 for (var i = 0; i < 5; i++) {2 (function(i) {3 setTimeout(function() {4 console.log(i);5 }, i * 1000);6 })(i);7 }
或者:
1 for(var i = 0; i < 5; i++) {2 setTimeout(function(i) {3 return function() {4 console.log(i);5 };6 }(i), i * 1000);7 }
输出:0 1 2 3 4
加上闭包,就能解决这个问题
4.如第一个假设删除了function(i)中的i呢,怎么办?
1 for (var i = 0; i < 5; i++) {2 (function() {3 setTimeout(function() {4 console.log(i);5 }, i * 1000);6 })(i);7 }
输出:5 5 5 5 5
内部其实没有对 i 保持引用,其实会变成输出 5个5
5.再更改一下
1 for (var i = 0; i < 5; i++) {2 setTimeout((function(i) {3 console.log(i);4 })(i), i * 1000);5 }
输出:0 1 2 3 4 (立刻输出,没有时间间隔)
给 setTimeout 传递了一个立即执行函数。setTimeout 可以接受函数或者字符串作为参数,那么这里立即执行函数是个啥呢,应该是个 undefined ,也就是说等价于:
setTimeout(undefined, ...);
而立即执行函数会立即执行,那么应该是立马输出的。
“应该是立马输出 0 到 4 吧。”
6.对于promise的考察
1 setTimeout(function() { 2 console.log(1) 3 }, 0); 4 new Promise(function executor(resolve) { 5 console.log(2); 6 for( var i=0 ; i<10000 ; i++ ) { 7 i == 9999 && resolve(); 8 } 9 console.log(3);10 }).then(function() {11 console.log(4);12 });13 console.log(5);
输出:2 3 5 4 1
考察 JavaScript 的运行机制的,
首先先碰到一个 setTimeout,于是会先设置一个定时,在定时结束后将传递这个函数放到任务队列里面,因此开始肯定不会输出 1 。
然后是一个 Promise,里面的函数是直接执行的,因此应该直接输出 2 3 。
然后,Promise 的 then 应当会放到当前 tick 的最后,但是还是在当前 tick 中。
因此,应当先输出 5,然后再输出 4 。
最后在到下一个 tick,就是 1 。
所以输出:2 3 5 4 1
一道容易栽坑的有趣的面试题(关于js,定时器,闭包等)