【javascript知识进阶】关于for循环中定义setTimeout
来源:互联网 发布:英雄联盟皮肤软件 编辑:程序博客网 时间:2024/06/06 14:15
for(var i = 0; i < 10; i++) { setTimeout(function() { console.log(i); }, 1000);}
上面的代码不会输出数字 0 到 9,而是会输出数字 10 十次,而且是同时输出。
可见,外部循环完毕后才对settimeout进行了定义,即i值变成10以后,所以实际上定义了十个i为10的timeout函数。
为什么呢?或许下面这个例子会让我们更直接的看到问题的答案:
var i = 10;setTimeout(function () { console.log(i)},0);i = 100;
输出结果是100。
页面中所有由setTimeout定义的操作,都将放在同一个队列中依次执行。
而这个队列执行的时间,需要等待到函数调用栈清空之后才开始执行。即所有可执行代码执行完毕之后,才会开始执行由setTimeout定义的操作。而这些操作进入队列的顺序,则由设定的延迟时间来决定。
了解了这些,上面的代码为什么出现这样的结果就很明显了:setTimeout定义时,所有的外部代码都执行完毕了,也就是说,在setTimeout所在的作用域中,其他代码会影响到我们所设定的时间值。
所以解决这个问题,我们只需要限制它的作用域就可以了,让这段作用域中只执行定义setTimeout的代码。
我们和可以用一个自执行函数来限制它的作用域,看起来就像给settimeout函数外部又“包”了一个函数,实际上这就形成了一个闭包。
其实广泛上来讲闭包判定的准则:即执行时是否在内部定义的函数中访问了上层作用域的变量。
for(var i = 0; i < 10; i++) { (function(e) { setTimeout(function() { console.log(e); }, 1000); })(i);}
这个自执行函数内部没有什么能改变变量i的代码了,所以进入该函数后settimeout就会马上被定义。外部的for循环无论怎么改变都不会影响传入该作用域的i值。
有另一个方法完成同样的工作,那就是从匿名包装器中返回一个函数。这和上面的代码效果一样。
for(var i = 0; i < 10; i++) { setTimeout((function(e) { return function() { console.log(e); } })(i), 1000)}
阅读全文
1 0
- 【javascript知识进阶】关于for循环中定义setTimeout
- 关于for循环中引用setTimeout
- 关于阿里的笔试题for循环中嵌套setTimeout
- js关于for循环里的setTimeout
- for循环中加入延时函数setTimeout
- js for循环中加settimeout问题
- for循环中setTimeout使用注意
- flex在for循环中使用setTimeout
- for循环中执行setTimeout问题
- for循环与setTimeout
- for循环和setTimeout
- for循环中的setTimeout()
- 【看书】关于for循环中定义变量的后续讨论
- Javascript中for循环
- for循环中定义变量
- 关于javascript中for/in循环的标签使用
- 关于For循环中定义的变量在循环外的引用
- 关于For循环中定义的变量在循环外的引用
- Servlet3.0
- Java SE、Java EE 与 Spring
- ArryList的继承实现关系
- (实用篇)PHP JSON数组与对象的理解
- 安卓窗口相关
- 【javascript知识进阶】关于for循环中定义setTimeout
- 剑指offer——二维数组中的查找
- Kotlin实现MVP设计模式
- 二叉树前序、中序、后序遍历非递归写法的透彻解析
- 解决highchart滚动条的问题
- 剑指Offer [03] 从尾到头打印链表
- JNI代码两种注册编写方式
- AC自动机详解
- Eclipse关闭XML文件验证的方法