详解JS中定时器setInterval和setTImeout的this指向问题


Posted in Javascript onJanuary 06, 2017

前言

Js是一个单线程语言,可以通过setTimeout()和setInterval()来设置代码在指定时刻运行,前者是在指定时间后执行,后者是指每隔一段时间执行。两者的使用方法类似。

最近在练习写一个小例子的时候用到了定时器,发现在setInterval和setTimeout中传入函数时,函数中的this会指向window对象,详细的介绍通过一个示例展开,一起来看看吧。

如下例:

var num = 0;
function Obj (){
 this.num = 1,
 this.getNum = function(){
 console.log(this.num);
 },
 this.getNumLater = function(){
 setTimeout(function(){
  console.log(this.num);
 }, 1000)
 }
}
var obj = new Obj; 
obj.getNum();//1打印的为obj.num,值为1
obj.getNumLater()//0
打印的为window.num,值为0

从上述例子中可以看到setTimeout中函数内的this是指向了window对象,这是由于setTimeout()调用的代码运行在与所在函数完全分离的执行环境上. 这会导致这些代码中包含的 this 关键字会指向 window (或全局)对象。详细可参考MDN setTimeout

但是在setTimeout中传入的不是函数时,this则指向当前对象,如下例:

var num = 0;
function Obj (){
 this.num = 1,
 this.getNum = function(){
 console.log(this.num);
 },
 this.getNumLater = function(){
 setTimeout(console.log(this.num), 1000)
 }
}
var obj = new Obj; 
obj.getNum();//1打印的为obj.num,值为1
obj.getNumLater()//1
打印的为obj.num,值为1

从以上两个例子可以看出,当在setTimeout中传入的参数为函数时,函数内部的this才会指向window对象。

当在setTimeout中传入了一个函数,若想要让this指向正确的值,可以使用以下两种比较常用的方法来使this指向正确的值:

1.将当前对象的this存为一个变量,定时器内的函数利用闭包来访问这个变量

如下:

var num = 0;
function Obj (){
 var that = this; //将this存为一个变量,此时的this指向obj
 this.num = 1,
 this.getNum = function(){
 console.log(this.num);
 },
 this.getNumLater = function(){
 setTimeout(function(){
  console.log(that.num); //利用闭包访问that,that是一个指向obj的指针
 }, 1000)
 }
}
var obj = new Obj; 
obj.getNum();//1打印的为obj.num,值为1
obj.getNumLater()//1
打印的为obj.num,值为1

这种方法是将当前对象的引用放在一个变量里,定时器内部的函数来访问到这个变量,自然就可以得到当前的对象。

2.利用bind()方法

var num = 0;
function Obj (){
 this.num = 1,
 this.getNum = function(){
 console.log(this.num);
 },
 this.getNumLater = function(){
 setTimeout(function(){
  console.log(this.num);
 }.bind(this), 1000) //利用bind()将this绑定到这个函数上
 }
}
var obj = new Obj; 
obj.getNum();//1打印的为obj.num,值为1
obj.getNumLater()//1
打印的为obj.num,值为1

bind()方法是在Function.prototype上的一个方法,当被绑定函数执行时,bind方法会创建一个新函数,并将第一个参数作为新函数运行时的this。在这个例子中,在调用setTimeout中的函数时,bind方法创建了一个新的函数,并将this传进新的函数,执行的结果也就是正确的了。关于bind方法可参考 MDN bind

以上两种方法都是比较常用的,当然如果使用call或apply方法来代替bind方法,得到的结果也是正确的,但是call方法会在调用之后立即执行,那样也就没有了延时的效果,定时器也就没有用了,所以推荐使用上述两种方法来将this传进setTimeout和setInterval中。

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作能带来一定的帮助,如果有疑问大家可以留言交流,谢谢大家对三水点靠木的支持。

Javascript 相关文章推荐
不同浏览器对回车提交表单的处理办法
Feb 13 Javascript
JQuery入门—JQuery程序的代码风格详细介绍
Jan 03 Javascript
javascript中数组中求最大值示例代码
Dec 18 Javascript
window resize和scroll事件的基本优化思路
Apr 29 Javascript
ionic 3.0+ 项目搭建运行环境的教程
Aug 09 Javascript
移动设备手势事件库Touch.js使用详解
Aug 18 Javascript
Angular2监听页面大小变化的解决方法
Oct 09 Javascript
使用Vue开发一个实时性时间转换指令
Jan 17 Javascript
vue toggle做一个点击切换class(实例讲解)
Mar 13 Javascript
vue利用axios来完成数据的交互
Mar 23 Javascript
js实现for循环跳过undefined值示例
Jul 02 Javascript
详解element-ui 表单校验 Rules 配置 常用黑科技
Jul 11 Javascript
Jqprint实现页面打印
Jan 06 #Javascript
JS使用正则截取两个字符串之间的字符串实现方法详解
Jan 06 #Javascript
jQuery EasyUi 验证功能实例解析
Jan 06 #Javascript
jQuery编写网页版2048小游戏
Jan 06 #Javascript
利用JQuery实现datatables插件的增加和删除行功能
Jan 06 #Javascript
javascript正则表达式模糊匹配IP地址功能示例
Jan 06 #Javascript
bootstrap导航栏、下拉菜单、表单的简单应用实例解析
Jan 06 #Javascript
You might like
php 获取完整url地址
2008/12/20 PHP
PHPMyadmin 配置文件详解(配置)
2009/12/03 PHP
PHP将HTML转换成文本的实现代码
2015/01/21 PHP
php中 $$str 中 "$$" 的详解
2015/07/06 PHP
php基于jquery的ajax技术传递json数据简单实例
2016/04/15 PHP
Yii编程开发常见调用技巧集锦
2016/07/15 PHP
网页设计常用的一些技巧
2006/12/22 Javascript
尽可能写"友好"的"Javascript"代码
2007/01/09 Javascript
JavaScript实现输入框(密码框)出现提示语
2016/01/12 Javascript
JavaScript 闭包详细介绍
2016/09/28 Javascript
原生JS实现图片翻书效果
2017/02/16 Javascript
jQuery插件echarts实现的去掉X轴、Y轴和网格线效果示例【附demo源码下载】
2017/03/04 Javascript
Angular4开发解决跨域问题详解
2017/08/28 Javascript
vue实现提示保存后退出的方法
2018/03/15 Javascript
JS实现计算小于非负数n的素数的数量算法示例
2019/02/26 Javascript
vue3.0中的双向数据绑定方法及优缺点
2019/08/01 Javascript
Nodejs技巧之Exceljs表格操作用法示例
2019/11/06 NodeJs
vue实现学生信息管理系统
2020/05/30 Javascript
vue+animation实现翻页动画
2020/06/29 Javascript
原生JS实现相邻月份日历
2020/10/13 Javascript
浅谈python中的getattr函数 hasattr函数
2016/06/14 Python
教你学会使用Python正则表达式
2017/09/07 Python
python使用原始套接字发送二层包(链路层帧)的方法
2019/07/22 Python
python GUI编程(Tkinter) 创建子窗口及在窗口上用图片绘图实例
2020/03/04 Python
html5使用canvas绘制文字特效
2014/12/15 HTML / CSS
瑞典最好的运动鞋专卖店:Sneakersnstuff
2016/08/29 全球购物
世界顶级户外运动品牌折扣网站:LeftLane Sports
2019/06/12 全球购物
为什么需要版本控制?
2013/08/08 面试题
教师自荐信范文
2013/12/09 职场文书
学生安全教育材料
2014/02/14 职场文书
2014年售后服务工作总结
2014/11/18 职场文书
网吧管理制度范本
2015/08/05 职场文书
培训感想范文
2015/08/07 职场文书
基于python制作简易版学生信息管理系统
2021/04/20 Python
全面盘点MySQL中的那些重要日志文件
2021/11/27 MySQL
openstack云计算keystone组件工作介绍
2022/04/20 Servers