等待this.$confirm执行完_JS 函数的执行时机

e61911df8481cf58d8a18d8aa465d978.png

下面代码为什么会打印出6个6

let i = 0
for(i = 0; i<6; i++){
  setTimeout(()=>{
    console.log(i)
  },0)
}

f4faeb741feb6f255e4e69904876326f.png

js执行机制与作用域链

首先,JavaScript是单线程环境,代码从上到下依次执行。这种执行方这也被称作是“同步执行”。(同一时间JavaScript只能执行一段代码,如果这段代码要执行很长时间,那么之后的代码只能尽情地等待它执行完才能有机会执行)。
但JavaScript中引进了异步机制。于是,所有任务可以分成两种,一种是同步任务(synchronous),另一种是异步任务(asynchronous)。同步任务指的是,在主线程上排队执行的任务,只有前一个任务执行完毕,才能执行后一个任务;异步任务指的是,不进入主线程、而进入”任务队列”(task queue)的任务,只有主线程上的任务执行完了,才通知”任务队列”,任务队列中的任务才会进入主线程执行。

在上面的代码中,for循环是同步代码,setTimeout是异步代码。遇到这种既包含同步又包含异步的情况,JavaScript依旧按照从上到下的顺序执行同步代码,并将异步代码插入任务队列。setTimeout的第二个参数则是把执行代码(console.log(i))添加到任务队列需等待的毫秒数,但等待的时间是相对主程序完毕的时间计算的,也就是说,在执行到setTimeout函数时会等待一段时间,再将当前任务插入任务队列。
最后,当执行完同步代码,js引擎就会去执行任务队列中的异步代码。这时候任务队列中就会有十个console.log(i)。我们知道,在每次循环中将setTimeout里面的代码“console.log(i)”放入任务队列时,i的值都是不一样的。但JavaScript引擎开始执行任务队列中的代码时,会开始在当前的作用域中开始找变量i,但是当前作用域中并没有对变量i进行定义。这个时候就会在创造该函数的作用域中寻找i。创建该函数的作用域就是全局作用域,这个时候就找到了for循环中的变量i,这时的i是全局变量,并且值已经确定:6。6个console.log“共享”i的值。这就是作用域链的问题。
为了解决以上两个问题,可以使用let或者闭包或者箭头函数。

打印0、1、2、3、4、5的方法

1、let方法

let i = 0
for(let i = 0; i < 6; i++){
  setTimeout(()=>{
    console.log(i)
  },0)
}

8fb4b14d05a8ad752c60b328c26787ef.png

2、闭包方法

for (i = 0; i < 6; i++) {
        !function(j){//闭包
            setTimeout(function(){
                console.log(j);
            },0)
        }(i);//闭包
    };

fda2bd46a43f391a012e2a9038bdf05f.png

引用文章:

用setTimeout实现for循环中的计时器​miss-me.github.io
f22a81f5611c773b767b800720925d1b.png

Just for myself

Amusing Ourselves to Death

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值