详解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 相关文章推荐
提高网站信任度的技巧
Oct 17 Javascript
ExtJS 2.0 实用简明教程之布局概述
Apr 29 Javascript
javascript检测是否联网的实现代码
Sep 28 Javascript
JavaScript闭包详解
Feb 02 Javascript
Jquery遍历select option和添加移除option的实现方法
Aug 26 Javascript
深入理解AngularJS中的ng-bind-html指令
Mar 27 Javascript
利用JS对iframe父子(内外)页面进行操作的方法教程
Jun 15 Javascript
解决vue-cli中stylus无法使用的问题方法
Jun 19 Javascript
react 国际化的实现代码示例
Sep 14 Javascript
vuex直接赋值的三种方法总结
Sep 16 Javascript
微信小程序实现简单跑马灯效果
May 26 Javascript
JavaScript canvas实现文字时钟
Jan 10 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
星际中一些鲜为人知的详细资料
2020/03/04 星际争霸
PHP实现链式操作的核心思想
2015/06/23 PHP
PHP强制转化的形式整理
2020/05/22 PHP
用js实现的一个Flash滚动轮换显示图片代码生成器
2007/03/14 Javascript
javascript div 遮罩层封锁整个页面
2009/07/10 Javascript
基本jquery的控制tabs打开的数量的代码
2010/10/17 Javascript
jQuery查询数据返回object和字符串影响原因是什么
2013/08/09 Javascript
JS+CSS实现的蓝色table选项卡效果
2015/10/08 Javascript
详解AngularJS中module模块的导入导出
2015/12/10 Javascript
整理一下常见的IE错误
2016/11/18 Javascript
Bootstrap基本组件学习笔记之面板(14)
2016/12/08 Javascript
JS实现的tab切换选项卡效果示例
2017/02/28 Javascript
jQuery ajax实现省市县三级联动
2021/03/07 Javascript
基于js 各种排序方法和sort方法的区别(详解)
2018/01/03 Javascript
Node.js模块全局安装路径配置方法
2018/05/17 Javascript
vue vue-Router默认hash模式修改为history需要做的修改详解
2018/09/13 Javascript
深入浅析Vue.js 中的 v-for 列表渲染指令
2018/11/19 Javascript
Vue 事件处理操作实例详解
2019/03/05 Javascript
微信小程序如何连接Java后台
2019/08/08 Javascript
对于Python的Django框架使用的一些实用建议
2015/04/03 Python
Python编程中用close()方法关闭文件的教程
2015/05/24 Python
Python的装饰器模式与面向切面编程详解
2015/06/21 Python
老生常谈python之鸭子类和多态
2017/06/13 Python
Python实现mysql数据库更新表数据接口的功能
2017/11/19 Python
PyQt5实现下载进度条效果
2018/04/19 Python
pycharm重置设置,恢复默认设置的方法
2018/10/22 Python
Python pycharm 同时加载多个项目的方法
2019/01/17 Python
Python Django的安装配置教程图文详解
2019/07/17 Python
pytorch实现保证每次运行使用的随机数都相同
2020/02/20 Python
浅析python 定时拆分备份 nginx 日志的方法
2020/04/27 Python
彪马日本官网:PUMA日本
2019/01/31 全球购物
什么是事务?事务有哪些性质?
2012/03/11 面试题
学院领导推荐信
2013/10/30 职场文书
彩妆大赛策划方案
2014/05/13 职场文书
放射科岗位职责
2015/02/14 职场文书
2019生态环境保护倡议书!
2019/07/03 职场文书