原文地址:→传送门

写在前面

setTimeout()是大家再熟悉不过的定时器,但平时对定时器的了解甚少,于是想看看setTimeout()的原理机制。

setTimeout()基础

setTimeout()函数用来指定某个函数或某段代码,在多少毫秒之后执行。它返回一个整数,表示定时器的编号,以后可以用来取消这个定时器。

var timeer = setTimeout(function|code,delay);

注:其中为code时需要以字符串形式传入,一般都是使用传入function的形式。

setTimeout(func,delay,arg1,arg2···)

setTimeout()可传入多个参数,第三个参数起是作为回调函数的参数传入。

        function add(a,b){
console.log(a+b);
}
setTimeout(add,4000,20,30); //20和30作为add的参数传入 结果为50

HTML5规定setTimeout()的最短时间间隔是4ms,对于不处在当前窗口的页面,浏览器会将时间间隔扩大到1000ms,若笔记本电脑处于电池供电状态,chrome及IE9以上版本,会将时间间隔切换到系统定时器,约15.6ms。

setTimeout()中回调函数的指向

setTimeout()回调函数调用的是某个对象的方法,则其中this指向的是setTimeout所处的环境而并非指向该对象环境。

可使用ES5中的bind()方法解决this指向,即将对象环境绑定到要执行的回调函数上即可。

        var name = 'lily';
function Person(name){
this.name = name;
this.printName = function(){
console.log(this.name);
}
} var person01 = new Person('jike'); //此处的this.name为window下的name,所以是'lily'
window.setTimeout(person01.printName,1000); //此处将printName绑定在person01上,所以this.name的值为'jike'
window.setTimeout(person01.printName.bind(person01),2000); //将printName中的this.name与person01限定在同一个作用域中,所以结果是'jike'
window.setTimeout(function(){person01.printName();},3000); //借助ES6中的箭头函数也可以解决问题
window.setTimeout(() => {person01.printName},1000);

setTimeout运行机制

setTimeout()和setInterval()的运行机制是,将指定的代码移出本次执行,等到下一轮Event Loop时,再检查是否到了指定时间。如果到了,就执行对应的代码;如果不到,就等到再下一轮Event Loop时重新判断。这意味着,setTimeout()指定的代码,必须等到本次执行的所有代码都执行完,才会执行。

例如:

task1();
setTimeout(otherTask,1000);
task2();

其中task1和task2为立即执行任务,otherTask被指定在1s后执行,此时otherTask被添加到任务队列的尾部,要等当前的脚本中Event Loop的任务队列全部执行完成后,才开始执行otherTask,但如果task1和task2很耗时,前面的任务超过1s还未结束,此时otherTask只能等task1和task2运行结束,才会执行otherTask。

setTimeout(func,0)的运用

当延迟时间设为0时,即当前任务队列一结束就立即执行func。(换个角度说就是同步任务的任务队列结束后尽早执行。)

        task01();
setTimeout(function(){
console.log('hello!'); //在task01和task02结束后立即打印hello!
},0)
task02();

0ms在实际上是达不到的,根据HTML5标准,setTimeout()推迟执行的时间最少是4ms,如果小于这个值,会被自动增加到4,同时也是为了防止多个setTimeout(func,0)语句连续执行,造成性能问题。

  • setTimeout(func,0)可以调整事件的发生顺序。

clearTimeout()

setTimeout()和setInterval()函数,都返回一个表示计数器编号的整数值,将该整数传入clearTimeout()和clearInterval()函数,就可以取消对应的定时器。两种定时器用的同一个编号池。

setTimeout()和setInterval()函数返回的整数值都是连续的,因此可以利用循环来清除所有的定时器。

防抖动(debounce)

该方法用于防止某个函数在短时间内被密集调用,具体来具体来说,debounce方法返回一个新版的该函数,这个新版函数调用后,只有在指定时间内没有新的调用,才会执行,否则就重新计时。

实际中不要设置太多个setTimeout()和setInterval(),它们耗费CPU,理想做法是将要延迟执行的代码都放在同一个函数里,然后支队这个函数使用setTimeout()和setInterval()。

setInterval()

setInterval()使用原理机制与setTimeout()一样,区别就是setInterval()是每隔多少ms执行一次回调函数。也可以传入大于两个参数。

setInterval()指定的是“开始执行”之间的间隔,并不考虑每次任指定的是“开始执行”之间的间隔,并不考虑每次任,比如,setInterval()指定每100ms执行一次,每次执行需要5ms,那么第一次执行后95ms后,第二次执行就会开始。如果某次执行需要105ms,即超过了delay时间,则执行结束后下一次执行就会立即开始。

setInterval()的最短间隔是10ms,小于10ms的时间间隔会被调整到10ms

setInterval()具有累积效应,如果某个操作特别耗时,超过了setInterval()的时间间隔,排在后面的操作就会被累积起来,然后在很短的时间内连续触发,这可能会造成性能问题。

资源参考

定时器(setTimeout)的秘密的更多相关文章

  1. 关于定时器 setTimeout

    1.这里不考虑线程问题.把javascript想象成在时间线上运行,页面载入时,首先执行的是<script>标签中的代码,之后,将执行更多代码,当进程空闲时,下个代码就被触发并执行 如图: ...

  2. Javascript 笔记与总结(2-13)定时器 setTimeout 和 setInterval

    定时器可以让 js 效果每隔几秒钟执行一次或者 n 秒之后执行某一个效果.定时器不属于 javascript,是 window 对象提供的功能. setTimeout 用法: window.setTi ...

  3. javascript定时器:setTimeout与setInterval

    概述: setTimeout:在指定的延迟时间之后调用一个函数或者执行一个代码片段,只执行一次: setInterval:周期性地调用一个函数(function)或者执行一段代码,重复执行: 语法格式 ...

  4. 定时器setTimeout()和Node.js的Event Loop

    一.定时器 setTimeout(fn,0)的含义是,指定某个任务在主线程最早可得的空闲时间执行,也就是说,尽可能早得执行.它在"任务队列"的尾部添加一个事件,因此要等到同步任务和 ...

  5. 关于定时器setTimeout()方法的实践--巧解bug

    _使用开发环境:UAP:_ _框架:JQuery.MX:_ 最近的开发的页面中,有一处需要在提交的 datagrid里启用行编辑,就会发生奇怪的bug,编辑过程中如图所示不移开焦点直接点保存,那么已输 ...

  6. 定时器 setTimeout()超时调用和 setInterval()间歇调用

    JavaScript是单线程语言,但它允许通过设置定时器,也就是设置超时值和间歇时间来调度代码在特定的时刻执行.前者是在指定的时间过后执行代码,而后者则是每隔指定的时间就执行一次代码. 超时调用需要使 ...

  7. 移动端长按响应事件以及阻止默认行为e.preventDefault()导致定时器setTimeout不能响应

    手指触摸绑定: $(document).on('touchstart', '.photo', function(e){ currentIndex = parseInt($(this).index('. ...

  8. 定时器setTimeout实现函数节流

    问题描述 文字输入查询的keyup或oninput事件,实现实时查询功能. 在用户输入过程中,用户可能只想需要 '小' 字的查询结果,但是以上两个事件会触发'x'.'i'.'a'.'o'.'小',一共 ...

  9. JavaScript之关闭轮询定时器(setTimeout/clearTimeout|setInterval/clearInterval)小结

    已知: 1.1 开启Timeout程序: scope.setTimeout("functionName()" | functionHandle, timeValue) 返回值:ti ...

  10. JS 定时器 setTimeout 与 setInterval 的区别和用法

    定时器: window.setTimeout(function(){},间隔时间毫秒); -- 定时炸弹,延迟执行,只执行一次 window.setInterval(function(){},间隔的时 ...

随机推荐

  1. 用hmmlearn学习隐马尔科夫模型HMM

    在之前的HMM系列中,我们对隐马尔科夫模型HMM的原理以及三个问题的求解方法做了总结.本文我们就从实践的角度用Python的hmmlearn库来学习HMM的使用.关于hmmlearn的更多资料在官方文 ...

  2. JAVAEE学习——struts2_03:OGNL表达式、OGNL与Struts2的结合和练习:客户列表

    一.OGNL表达式 1.简介 OGNL:对象视图导航语言.  ${user.addr.name} 这种写法就叫对象视图导航. OGNL不仅仅可以视图导航.支持比EL表达式更加丰富的功能. 2.使用OG ...

  3. 用PHP和Ajax进行前后台数据交互——以用户登录为例

    很多网站中都有用户登录系统,要完成用户的注册和登陆,就一定要用到前后台的数据交互.在这里以简单的用户注册和登陆为例介绍一下前后台交互的大致流程. 首先,我们来做一个简单的登陆界面. 这里为了方便我使用 ...

  4. AngularJS高级程序设计读书笔记 -- 控制器篇

    作用域组成了一个能够用于在控制器之间形成通信的体系结构. 1. 控制器和作用域的基本原理 控制器就像领域模型与视图之间的纽带, 他给视图提供数据与服务, 并且定义了所需的业务逻辑, 从而将用户行为转换 ...

  5. 使用requireJs的方法

    在你们对requireJs初步了解后,快来看看他们是怎么使用的吧. 在你下载完成require.js插件后,在页面里引入,在require.js 加载完之后,会查找页面上script标签的data-m ...

  6. [编织消息框架][netty源码分析]8 Channel 实现类NioSocketChannel职责与实现

    Unsafe是托委访问socket,那么Channel是直接提供给开发者使用的 Channel 主要有两个实现 NioServerSocketChannel同NioSocketChannel 致于其它 ...

  7. 按bean的属性值对list集合进行排序

    List根据对象的某个属性排序工具类 List排序 import java.util.Collections; import java.util.Comparator; import java.uti ...

  8. linux开机启动smb服务

    修改/etc/rc.local文件(增加红色部分) [root@localhost ~]# cat /etc/rc.local #!/bin/sh## This script will be exec ...

  9. Spring Boot快速入门

    安装 安装依赖 maven是一个依赖管理工具,我们利用maven进行构建.创建一个maven项目,在pom.xml里面添加依赖项 <?xml version="1.0" en ...

  10. 【转】JS容器拖拽效果,并通过cookie保存拖拽各容器的所在位置

    <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...