详解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 相关文章推荐
JavaScript 学习笔记之操作符
Jan 14 Javascript
使用jQuery实现input数值增量和减量的方法
Jan 24 Javascript
window.onload与$(document).ready()的区别分析
May 30 Javascript
实例讲解jquery与json的结合
Jan 07 Javascript
AngularJS基础 ng-list 指令详解及示例代码
Aug 02 Javascript
前端编码规范(3)JavaScript 开发规范
Jan 21 Javascript
详解Javascript获取缓存和清除缓存API
May 25 Javascript
Vue的Flux框架之Vuex状态管理器
Jul 30 Javascript
AngularJS中下拉框的高级用法示例
Oct 11 Javascript
JS伪继承prototype实现方法示例
Jun 20 Javascript
小程序跨页面交互的作用与方法详解
Jan 07 Javascript
JS实现碰撞检测效果
Mar 12 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
使用Discuz关键词服务器实现PHP中文分词
2014/03/11 PHP
Ajax+PHP实现的删除数据功能示例
2019/02/12 PHP
PHP进阶学习之垃圾回收机制详解
2019/06/18 PHP
基于jquery的从一个页面跳转到另一个页面的指定位置的实现代码(带平滑移动的效果)
2011/05/24 Javascript
Javascript 面向对象编程(一) 封装
2011/08/28 Javascript
利用JS自动打开页面上链接的实现代码
2011/09/25 Javascript
js 弹出菜单/窗口效果
2011/10/30 Javascript
扩展IE中一些不兼容的方法如contains、startWith等等
2014/01/09 Javascript
jquery选择器排除某个DOM元素的方法(实例演示)
2014/04/25 Javascript
详解 javascript中offsetleft属性的用法
2015/11/11 Javascript
jQuery Ajax 实例代码 ($.ajax、$.post、$.get)
2016/04/29 Javascript
js原生实现FastClick事件的实例
2016/11/20 Javascript
Javascript 数组去重的方法(四种)详解及实例代码
2016/11/24 Javascript
jquery pagination分页插件使用详解(后台struts2)
2017/01/22 Javascript
JS跨域请求外部服务器的资源
2017/02/06 Javascript
基于canvas的二维码邀请函生成插件
2017/02/14 Javascript
Bootstrap输入框组件使用详解
2017/06/09 Javascript
关于vue-router的beforeEach无限循环的问题解决
2017/09/09 Javascript
vue .sync修饰符的使用详解
2018/06/15 Javascript
解决vuejs 使用value in list 循环遍历数组出现警告的问题
2018/09/26 Javascript
详解VUE中的插值( Interpolation)语法
2020/10/18 Javascript
python模块之paramiko实例代码
2018/01/31 Python
Python使用matplotlib绘图无法显示中文问题的解决方法
2018/03/14 Python
解决pycharm工程启动卡住没反应的问题
2019/01/19 Python
python 哈希表实现简单python字典代码实例
2019/09/27 Python
python可视化text()函数使用详解
2020/02/11 Python
python如何把字符串类型list转换成list
2020/02/18 Python
Python使用正则表达式实现爬虫数据抽取
2020/08/17 Python
浅谈anaconda python 版本对应关系
2020/10/07 Python
幼儿园中秋节活动方案
2014/02/06 职场文书
投资意向书范本
2014/04/01 职场文书
家长写给孩子的评语
2014/04/18 职场文书
实习生评语
2014/04/26 职场文书
自主招生教师推荐信
2014/05/10 职场文书
学习党的群众路线剖析材料
2014/10/09 职场文书
再谈python_tkinter弹出对话框创建
2022/03/20 Python