• 一道经典面试题-----setTimeout(function(){},0)


    出处:http://www.qdfuns.com/notes/17398/e8a1ce8f863e8b5abb530069b388a158/page/3.html#tagsbar

    1 for (var i = 0; i < 3; i++) {
    2     setTimeout(function() {
    3         console.log(i);
    4     }, 0);
    5     console.log(i);
    6 }

    效果:

    console.log: 0
    console.log: 1
    console.log: 2
     
    结果是:0 1 2 3 3 3
    很多公司面试都爱出这道题,此题考察的知识点还是蛮多的。
    为了防止初学者栽在此问题上,此文稍微分析一下。
    都考察了那些知识点呢?
    异步、作用域、闭包,你没听错,是闭包。
    我们来简化此题:
    1 setTimeout(function() {
    2         console.log(1);
    3 }, 0);
    4 console.log(2);

    先打印2,后打印1。
    因为是setTimeout是异步的。
    正确的理解setTimeout的方式(注册事件):
    有两个参数,第一个参数是函数,第二参数是时间值。
    调用setTimeout时,把函数参数,放到事件队列中。等主程序运行完,再调用。
    没啥不好理解的。就像我们给按钮绑定事件一样:

    1 btn.onclick = function() {
    2         alert(1);
    3 };

    这么写完,会弹出1吗。不会!!只是绑定事件而已!
    必须等我们去触发事件,比如去点击这个按钮,才会弹出1。
    setTimeout也是这样的!只是绑定事件,等主程序运行完毕后,再去调用。
    setTimeout的时间值是怎么回事呢?
    比如:

    1 setTimeout(fn, 2000)

    我们可以理解为2000之后,再放入事件队列中,如果此时队列为空,那么就直接调用fn。如果前面还有其他的事件,那就等待。
    因此setTimeout是一个约会从来都不准时的童鞋。

    继续看:

    1 setTimeout(function() {
    2         console.log(i);
    3 }, 0);
    4 var i = 1;

    程序会不会报错?
    不会!而且还会准确得打印1。
    为什么?
    因为真正去执行console.log(i)这句代码时,var i = 1已经执行完毕了!
    所以我们进行dom操作。可以先绑定事件,然后再去写其他逻辑。

    1 window.onload = function() {
    2         fn();
    3 }
    4 var fn = function() {
    5         alert('hello')
    6 };

    这么写,完全是可以的。因为异步!

    es5中是没有块级作用域的

    1 for (var i = 0; i < 3; i++) {}
    2 console.log(i);

    也就说i可以在for循环体外访问到。所以是没有块级作用域。
    但此问题在es6里终结了,因为es6,发明了let。

    这回我们再来看看原题。
    原题使用了for循环。循环的本质是干嘛的?
    是为了方便我们程序员,少写重复代码。
    让我们倒退50年,原题等价于:

     1 var i = 0;
     2 setTimeout(function() {
     3     console.log(i);
     4 }, 0);
     5 console.log(i);
     6 i++;
     7 setTimeout(function() {
     8     console.log(i);
     9 }, 0);
    10 console.log(i);
    11 i++;
    12 setTimeout(function() {
    13     console.log(i);
    14 }, 0);
    15 console.log(i);
    16 i++;

    结果:

    console.log: 0
    console.log: 1
    console.log: 2
     
    因为setTimeout是注册事件。根据前面的讨论,可以都放在后面。
    原题又等价于如下的写法:
     1 var i = 0;
     2 console.log(i);
     3 i++;
     4 console.log(i);
     5 i++;
     6 console.log(i);
     7 i++;
     8 setTimeout(function() {
     9     console.log(i);
    10 }, 0);
    11 setTimeout(function() {
    12     console.log(i);
    13 }, 0);
    14 setTimeout(function() {
    15     console.log(i);
    16 }, 0);

    结果:

    console.log: 0
    console.log: 1
    console.log: 2
    这回你明白了为啥结果是0 1 2 3 3 3了吧。

    那个,老姚你说它是闭包,又是怎么回事?
    为了很好的说明白这个事情,我们把它放到一个函数中:
    1 var fn = function() {
    2         for (var i = 0; i < 3; i++) {
    3                 setTimeout(function() {
    4                         console.log(i);
    5                 }, 0);
    6                 console.log(i);
    7         }
    8 };
    9 fn();

    上面的函数跟我们常见另一个例子(div绑定事件)有什么区别:

    1 var fn = function() {
    2         var divs = document.querySelectorAll('div');
    3         for (var i = 0; i < 3; i++) {
    4                 divs[i].onclick = function() {
    5                         alert(i);
    6                 };
    7         }
    8 };
    9 fn();

    点击每个div都会弹出3。道理是一样的。因为alert(i)中的i是fn作用越中的,因而这是闭包。
    《javascript忍者秘籍》书里把一个函数能调用全局变量,也称闭包。
    因为作者认为全局环境也可以想象成一个大的顶级函数。
    怎么保证能弹出0,1, 2呢。
    解决之道:以毒攻毒!
    再创建个闭包!!

     1 var fn = function() {
     2         var divs = document.querySelectorAll('div');
     3         for (var i = 0; i < 3; i++) {
     4                 divs[i].onclick = (function(i) {
     5                         return function() {
     6                                 alert(i);
     7                         };
     8                 })(i);
     9         }
    10 };
    11 fn();

    或者如下的写法:

     1 var fn = function() {
     2         var divs = document.querySelectorAll('div');
     3         for (var i = 0; i < 3; i++) {
     4                 (function(i) {
     5                         divs[i].onclick = function() {
     6                                 alert(i);
     7                         };
     8                 })(i);
     9         }
    10 };
    11 fn();

    因此原题如果也想setTimeout也弹出0,1,2的话,改成如下:

    1 for (var i = 0; i < 3; i++) {
    2     setTimeout((function(i) {
    3         return function() {
    4             console.log(i);
    5         };
    6     })(i), 0);
    7     console.log(i);
    8 }

    结果:

    console.log: 0
    console.log: 1
    console.log: 2
  • 相关阅读:
    ES6 常用总结(前端开发js技术进阶提升总结)
    web前端之es6对象的扩展
    ES6数组及对象遍历的新增方法 entries(),keys() 和 values()
    关于日期
    最近遇到的几个小东西
    求模
    同步 异步请求的认识
    变量名和函数名声明提升
    机顶盒前端开发小结
    js节点使用 碎片节点
  • 原文地址:https://www.cnblogs.com/thelongmarch/p/7645144.html
Copyright © 2020-2023  润新知