为了账号安全,请及时绑定邮箱和手机立即绑定

前端面试的一个小问题

前端面试的一个小问题

月关宝盒 2018-08-01 22:38:00
昨天去一家公司面试前端,第二轮技术面给我出了一个题var len=4;while(len--){     setTimeout(function(){         alert(len);     },0);     alert(len); }问输出结果当时有点蒙,自己不熟悉setTimeOut函数以为time为0就是立即调用,然后考官说setTimeOut是异步的,即使是0也会在最后执行,结果输出应该为3,2,1,0,-1,-1,-1,-1.后来又变了题目是:var len=4;while(len--){     (function(i){         setTimeout(function(){             alert(i);         },0);     })(len);     alert(len); }这次我就很清楚了,产生一个闭包,每次len的变化i都保存了一个copy,所以输出是3,2,1,0,3,2,1,0.但是刚才写demo的时候发现第一段代码的输出符合预期,第二段的输出开始4个是3,2,1,0但是后面的也就是setTimeout中的输出有时候都不一样,有时是0,2,1,3有时是2,1,0,3等等求解释。
查看完整描述

2 回答

?
精慕HU

TA贡献1845条经验 获得超8个赞

其实第二个题目的答案确实一定是:32103210

之所以后面四个 alert 输出 的结果有不确定性,和 setTimeout 以及延迟时间 都没有关系,是因为,alert 会阻塞浏览器的执行线程,而所有阻塞执行线程的方式,都会让setTimeout的结果具有不确定性。

(即便setTimeout的延迟时间为0,回调函数也是严格的压入队列栈中的,按照FIFO的顺序依次调用。)

比如,打开调试工具,设置断点,如果执行到延时回调函数中断点生效了,这时,代码中的 setTimeout setInterval 等结果也都会受到影响。

遇到这种情况,最好的方式(比如此处),尽量使用【console.log】输出结果,这样只会将对象打印到控制台,而不会影响时间性和顺序。


查看完整回答
反对 回复 2018-08-05
?
qq_花开花谢_0

TA贡献1835条经验 获得超7个赞

前边那个你理解了?那我只说后边那个——闭包。

 (function(i){ setTimeout(function(){ alert(i); },0); })(len);

这里创建了一个函数并立即调用了,函数的参数 i 被设置成 len 现在的值,然后它就不改了。所以它 alert 的值是 0-3 而不是 -1。

至于顺序,都是延时 0 毫秒执行,所以顺序是乱的。


查看完整回答
反对 回复 2018-08-05
  • 2 回答
  • 0 关注
  • 989 浏览
慕课专栏
更多

添加回答

举报

0/150
提交
取消
意见反馈 帮助中心 APP下载
官方微信