详解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 相关文章推荐
禁止刷新,回退的JS
Nov 25 Javascript
jquery中ajax调用json数据的使用说明
Mar 17 Javascript
javascript之typeof、instanceof操作符使用探讨
May 19 Javascript
随窗体滑动的小插件sticky源码
Jun 21 Javascript
JavaScript中伪协议 javascript:使用探讨
Jul 18 Javascript
node.js中的path.resolve方法使用说明
Dec 08 Javascript
JavaScript计时器示例分析
Feb 05 Javascript
简述JavaScript对传统文档对象模型的支持
Jun 16 Javascript
JavaScript的Polymer框架中dom-repeat与VM的相关操作
Jul 29 Javascript
Bootstrap按钮下拉菜单组件详解
May 10 Javascript
JS实现中文汉字按拼音排序的方法
Oct 09 Javascript
javascript原型链学习记录之继承实现方式分析
May 01 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中显示格式化的用户输入
2006/10/09 PHP
浅析PHP 按位与或 (^ 、&)
2013/06/21 PHP
php单态设计模式(单例模式)实例
2014/11/18 PHP
PHP中读取文件的几个方法总结(推荐)
2016/06/03 PHP
PHP中的正则表达式实例详解
2017/04/25 PHP
PHP receiveMail实现收邮件功能
2018/04/25 PHP
PHP基于进程控制函数实现多线程
2020/12/09 PHP
js中访问html中iframe的文档对象的代码[IE6,IE7,IE8,FF]
2011/01/08 Javascript
Js 导出table内容到Excel的简单实例
2013/11/19 Javascript
纯js实现重发验证码按钮倒数功能
2015/04/21 Javascript
Javascript实现的SHA-256加密算法完整实例
2016/02/02 Javascript
jQuery检查元素存在性(推荐)
2016/09/17 Javascript
用原生js做单页应用
2017/01/17 Javascript
vue与TypeScript集成配置最简教程(推荐)
2017/10/17 Javascript
解决微信小程序防止无法回到主页的问题
2018/09/28 Javascript
使用pm2自动化部署node项目的方法步骤
2019/01/28 Javascript
vue路由中前进后退的一些事儿
2019/05/18 Javascript
微信小程序实现加入购物车滑动轨迹
2020/11/18 Javascript
Python 元类使用说明
2009/12/18 Python
Python打印斐波拉契数列实例
2015/07/07 Python
python实现简易云音乐播放器
2018/01/04 Python
Html5 Geolocation获取地理位置信息实例
2016/12/09 HTML / CSS
巴西备受欢迎的服装和生活方式品牌:FARM Rio
2020/02/04 全球购物
物流专业求职计划书
2014/01/10 职场文书
好邻里事迹材料
2014/01/16 职场文书
幼儿教师考核制度
2014/01/25 职场文书
实习指导老师评语
2014/04/26 职场文书
员工生日会策划方案
2014/06/14 职场文书
2014镇党委班子对照检查材料思想汇报
2014/09/23 职场文书
入党积极分子自我批评思想汇报
2014/10/10 职场文书
国家助学金感谢信
2015/01/21 职场文书
工作能力自我评价2015
2015/03/05 职场文书
恰同学少年观后感
2015/06/08 职场文书
2016年感恩节寄语
2015/12/07 职场文书
检讨书怎么写?
2019/06/21 职场文书
合理缓解职场压力,让你随时保持最佳状态!
2019/06/21 职场文书