setTimeout和setInterval的区别

setTimeout和setInterval这两个函数, 你们确定都不陌生, 但可能并非每一个人都了解其内部的实质。javascript

甚至可能会错误的把两个实现定时调用的函数理解成了相似thread同样的东西, 认为会在一个时间片内, 并发的执行调用的函数, 彷佛很好很强大。但其实并非如此, 实际的状况是javascript都是以单线程的方式运行于浏览器的javascript引擎中的。java

setTimeout和setInterval的做用只是把你要执行的代码在你设定的一个时间点插入js引擎维护的一个代码队列中, 插入代码队列并不意味着你的代码就会立马执行的,理解这一点很重要. 并且setTimeout和setInterval还有点不同。浏览器

 

1、setTimeout

function click() {
  // code block1...
  setTimeout(function() {
    // process ...
  }, 200);
  // code block2
} 

假设咱们给一个button的onclick事件绑定了此方法。并发

当咱们按下按钮后, 确定先执行block1的内容, 而后运行到setTimeout的地方, setTimeout会告诉浏览器说: "200ms后我会插一段要执行的代码给你的队列中", 浏览器固然答应了(注意插入代码并不意味着立马执行), setTimeout代码运行后, 紧跟其后的block2代码开始执行。函数

这里就开始说明问题了, 若是block2的代码执行时间超过200ms, 那结果会是如何? 或许按照你以前的理解, 会理所固然的认为200ms一到, 你的process代码会立马执行...事实是, 在block2执行过程当中(执行了200ms后)process代码被插入代码队列, 但一直要等click方法执行结束, 才会执行process代码段, 从代码队列上看process代码是在click后面的, 再加上js以单线程方式执行, 因此应该不难理解. 若是是另外一种状况, block2代码执行的时间<200ms, setTimeout在200ms后将process代码插入到代码队列, 而那时执行线程可能已经处于空闲状态了(idle), 那结果就是200ms后, process代码插入队列就立马执行了, 就让你感受200ms后, 就执行了。spa

 

2、setInterval

这里可能会存在两个问题:
1.时间间隔或许会跳过
2.时间间隔可能小于定时调用的代码的执行时间线程

function click() {
  
// code block1...   setInterval(function() {     // process ...   }, 200);   // code block2 }

和上面同样咱们假设经过一个click, 触发了setInterval以实现每隔一个时间段执行process代码code

好比:onclick要300ms执行完, block1代码执行完, 在5ms时执行setInterval, 以此为一个时间点, 在205ms时插入process代码, click代码顺利结束, process代码开始执行(至关于图中的timer code), 然而process代码也执行了一个比较长的时间, 超过了接下来一个插入时间点405ms, 这样代码队列后又插入了一份process代码, process继续执行着, 并且超过了605ms这个插入时间点。blog

下面问题来了, 可能你还会认为代码队列后面又会继续插入一份process代码...... 真实的状况是,因为代码队列中已经有了一份未执行的process代码, 因此605ms这个插入时间点将会被"无情"的跳过, 由于js引擎只容许有一份未执行的process代码, 说到这里不知道您是否是会豁然开朗呢...队列

为了解决这种问题你能够用一种更好的代码形式:

setTimeout(function(){
  //processing
  setTimeout(arguments.callee, interval);
}, interval); 

这个估计稍微想一下, 就明白其中的好处了, 这样就不会产生时间点被跳过的问题了。

相关文章
相关标签/搜索