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

发布时间 - 2026-01-10 22:21:01    点击率:

前言

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中。

总结

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


# setinterval  # this  # settimeout  # js  # this指向问题  # javascript中SetInterval与setTimeout的定时器用法  # JavaScript定时器setTimeout()和setInterval()详解  # 定时器(setTimeout/setInterval)调用带参函数失效解决方法  # 理解javascript定时器中的setTimeout与setInterval  # Javascript中setTimeOut和setInterval的定时器用法  # JavaScript中定时器setTimeout()和setInterval()的用法  # JavaScript定时器setTimeout、setInterval使用详解  # 值为  # 两种  # 是一个  # 是在  # 也就  # 并将  # 方法来  # 绑定  # 都是  # 这是  # 放在  # 在这个  # 第一个  # 才会  # 是指  # 推荐使用  # 会在  # 可以通过  # 用了  # 可以看到 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: Laravel如何设置定时任务(Cron Job)_Laravel调度器与任务计划配置  如何快速生成专业多端适配建站电话?  网站制作软件免费下载安装,有哪些免费下载的软件网站?  音响网站制作视频教程,隆霸音响官方网站?  如何选择PHP开源工具快速搭建网站?  javascript和jQuery中的AJAX技术详解【包含AJAX各种跨域技术】  如何为不同团队 ID 动态生成多个独立按钮  Win11搜索栏无法输入_解决Win11开始菜单搜索没反应问题【技巧】  如何在建站主机中优化服务器配置?  免费网站制作appp,免费制作app哪个平台好?  如何用5美元大硬盘VPS安全高效搭建个人网站?  C语言设计一个闪闪的圣诞树  JavaScript数据类型有哪些_如何准确判断一个变量的类型  百度浏览器如何管理插件 百度浏览器插件管理方法  Laravel怎么实现模型属性转换Casting_Laravel自动将JSON字段转为数组【技巧】  如何在阿里云香港服务器快速搭建网站?  Windows驱动无法加载错误解决方法_驱动签名验证失败处理步骤  如何在万网ECS上快速搭建专属网站?  百度浏览器网页无法复制文字怎么办 百度浏览器复制修复  米侠浏览器网页背景异常怎么办 米侠显示修复  Laravel如何实现多语言支持_Laravel本地化与国际化(i18n)配置教程  动图在线制作网站有哪些,滑动动图图集怎么做?  如何挑选最适合建站的高性能VPS主机?  Mybatis 中的insertOrUpdate操作  百度输入法ai面板怎么关 百度输入法ai面板隐藏技巧  阿里云高弹*务器配置方案|支持分布式架构与多节点部署  敲碗10年!Mac系列传将迎来「触控与联网」双革新  高端建站如何打造兼具美学与转化的品牌官网?  微博html5版本怎么弄发超话_超话进入入口及发帖格式要求【教程】  Laravel如何将应用部署到生产服务器_Laravel生产环境部署流程  Claude怎样写约束型提示词_Claude约束提示词写法【教程】  Firefox Developer Edition开发者版本入口  深圳网站制作的公司有哪些,dido官方网站?  香港代理服务器配置指南:高匿IP选择、跨境加速与SEO优化技巧  Laravel如何生成PDF或Excel文件_Laravel文档导出工具与使用教程  Laravel如何与Vue.js集成_Laravel + Vue前后端分离项目搭建指南  香港服务器选型指南:免备案配置与高效建站方案解析  详解Android图表 MPAndroidChart折线图  Laravel如何创建和注册中间件_Laravel中间件编写与应用流程  Laravel 419 page expired怎么解决_Laravel CSRF令牌过期处理  b2c电商网站制作流程,b2c水平综合的电商平台?  标题:Vue + Vuex 项目中正确使用 JWT 进行身份认证的实践指南  Laravel如何配置任务调度?(Cron Job示例)  如何用PHP快速搭建高效网站?分步指南  如何在宝塔面板创建新站点?  Laravel如何使用Sanctum进行API认证?(SPA实战)  android nfc常用标签读取总结  Java遍历集合的三种方式  使用spring连接及操作mongodb3.0实例  如何在万网主机上快速搭建网站?