首页 > 其他分享 >【js】for 循环中使用 setTimeout 的问题

【js】for 循环中使用 setTimeout 的问题

时间:2022-09-01 21:23:29浏览次数:93  
标签:setTimeout 作用域 js 队列 任务 循环 var 执行

问题:
下面代码的输出结果不是间隔3秒依次输出 1, 2, 3, 4, 5。而是隔了3秒连续输出6。这是为什么呢?

for (var i = 1; i <= 5; i++){
    setTimeout(function timer() {
      console.log(i)
    }, 3000);
  }

先说JS的执行机制和作用域
首先,JavaScript是单线程环境,代码从上到下依次执行。这种执行方式被称作为是“同步执行”。

但是,JavaScript引进了异步执行机制,也就是事件循环机制。所以,任务可以分为两种:一种是同步任务;另一种是异步任务。同步任务是指:在主线程上排队执行的任务,只有前一个任务执行完毕,才能执行后一个任务。异步任务是指:不进入主线程,而进入任务队列的任务,只有当主线程上的任务执行完毕后,才通知任务队列,任务队列中的任务才会进入主线程执行。

问题中的代码:for循环是同步代码,setTimeout是异步代码。在这种情况下,JavaScript依然按照从上到下执行先执行同步的代码,并将异步的代码放入到任务队列中。setTimeout的第二个参数是把执行代码console.log(i)添加到任务队列所需要的毫秒数,等待的时间是相对主程序执行完毕的时间计算的。也就是说,在主程序执行完后,会等待一会时间,再将setTimeout任务插入到任务队列中。

这样,在执行完同步的代码后,再去执行任务队列中的异步代码。这个时候,任务队列中会有5个console.log(i)等待执行。起初,以为每次循环,setTimeout放入到console.log(i)中的i的值是不一样的,但是,JavaScript引擎在开始执行任务队列中代码时,会在当前作用域中寻找变量i,但是当前作用域(也就是setTimeout中)并没有对 i 的定义。这个时候就会从创建该函数的作用域中寻找变量 i,创建该函数的作用域是全局作用域,所以就找到了for循环中的变量 i ,而这时的全局变量 i 的值已经确定为:6.所以最后打印出来的是5个6

解决方案一:

for (var i = 1; i <= 5; i++){
    setTimeout(function timer(i) {
      console.log(i)
    }(i), 3000);
  }

这样会发现打印出来的值是1, 2, 3, 4, 5;但是没有时间间隔就输出。这时因为在function timer()后面添加()表示的是立即执行函数,只要setTimeout调用就执行了,而不是等到定时器到后才执行

优化:

for (var i = 1; i <= 5; i++) {
    (function (j) {
      setTimeout(function timer() {
        console.log(j)
      }, 3000);     
    })(i)
  }

这样会发现打印出来的值是1, 2, 3, 4, 5;会等待3秒再同时输出;与上面不同,这次的setTimeout要等到定时器到后才能执行。

优化:

如果想要每次输出后间隔一秒,只需要将传入的 i 和 time 相乘即可:

 for (var i = 1; i <= 5; i++) {
    (function (j) {
      setTimeout(function timer() {
        console.log(j)
      }, j * 1000);     
    })(i)
  }

至于为什么是每隔一秒?因为按照上面写法,每隔setTimeout的定时器依次是 1s,2s,3s,4s,5s。之前说过,定时器计时是相对于主线程运行完毕后开始的,所以相邻输出间隔1s.

解决方案二:拆分结构

 function timer(i) {
    setTimeout(() => {
      console.log(i)
    }, i *1000);
  }

  for (var i = 1; i <= 5; i++){
    timer(i)
  }

也是解决了 i 的作用域问题。输出也是每隔一秒输出。

解决方案三:let
let声明的变量只在其声明的块或子块中可用,这一点,与var相似。二者之间最主要的区别在于var声明的变量的作用域是整个封闭函数。

for (let i = 1; i <= 5; i++){
    setTimeout(function timer() {
      console.log(i)
    }, i * 1000);
  }

这样打印的效果也是每隔一秒打印一个数值:1, 2, 3, 4, 5.

解决方案四:用new Promise实现一个delay

function delay(ms) {
    return new Promise((resolve, reject) => {
        setTimeout(() => {
            resolve()
        }, ms);
    })
}
for (let i = 1; i <= 5; i++) {
    await delay(1000)
    console.log(i)
}

这样结构更清晰,delay也可以复用

标签:setTimeout,作用域,js,队列,任务,循环,var,执行
From: https://www.cnblogs.com/vickylinj/p/16647850.html

相关文章

  • native <-互相调用-> js
    WebViewWebView提供了这些方法-提供了这些方法```addJavascriptInterfacecanGoBackcanGoBackOrForwardcanGoForwardcapturePictureclearCacheclearFormData......
  • 2022-8-31 jsp el表达式
    jsp<%--JSP脚本片段:用于在JSP页面写java代码--%>注意:1、JSP脚本片段中只能出现java代码,不能出现HTML元素。在 访问JSP时,JSP引擎翻译JSP页面中的......
  • 【js与native通信】1 通信协议制定
    通过native<-互相调用->js知道WebView有一个方法setWebChromeClient,可以设置WebChromeClient对象。而WebChromeClient对象中有三个方法,分别是onJsAlert......
  • js 实现冒泡排序及优化方案
    //冒泡排序//原理就是每一轮循环,将一个最大的值放冒泡到最后//1.每一趟都是比较相邻两个元素,如果后一个元素大于前一个,则交换两个元素//2.第一趟从第一个元素开始......
  • 人均瑞数系列,瑞数 5 代 JS 逆向分析
    声明本文章中所有内容仅供学习交流使用,不用于其他任何目的,不提供完整代码,抓包内容、敏感网址、数据接口等均已做脱敏处理,严禁用于商业用途和非法用途,否则由此产生的一切......
  • mysql 储存过程 如何使用递归循环来实现sql数据恢复
    CREATEPROCEDURE`relation_update`(inePIDvarchar(100),indeptidint)BEGINDECLAREtuivarchar(100);declarectint;declareiint;DECLAREcur1CURSORFORselect......
  • JS 事件盘点
    1.鼠标事件鼠标在元素上移动mousemove鼠标从该元素上移出mouseout鼠标移动到该元素mouseover鼠标点击(单击)click鼠标双击dbclick鼠标右击contextmenu可以配......
  • 生成二维码并以图片格式下载-qrcodejs2
    1、安装qrcodejs2npminstallqrcodejs2--save2、在需要的页面引入importQRCodefrom"qrcodejs2";3、页面中使用<divid="qrcode"ref="qrcode"></div>4......
  • Flask 学习-31.flask_jwt_extended 验证token四种方headers/cookies/json/query_stri
    前言用户携带授权token访问时,其jwt的所处位置列表,默认是在请求头部headers中验证。可以通过JWT_TOKEN_LOCATION进行全局配置,设置token是在请求头部,还是cookies,还是json,......
  • JS缓存三种方法
    1.sessionStorage:临时的会话存储只要当前的会话窗口未关闭,存储的信息就不会丢失,即便刷新了页面,或者在编辑器中更改了代码,存储的会话信息也不会丢失。2.localStor......