事件循环是什么东西,一篇文章带你了解

为什么事件会循环?

因为js是默认单线程执行的,但是在执行过程中往往会有耗时任务和不耗时任务,为了效率,js往往会先把不耗时任务(同步任务)执行完,再执行耗时任务(异步任务)

既然如此,为什么不设置多线程呢?

举个例子:如果两个线程同时遇见了一个a=1,另外一个a=2呢,那么他们该怎么执行?往往就会报错,为了解决这种问题,程序员又需要在程序当中添加锁,就会增加代码难度,也会增加设备性能开销

任务执行顺序

在执行的过程中,js通常是从上往下执行,遇到了异步任务的时候,会把任务放到队列中,等同步任务全部结束,再从队列中按照先进先出的顺序,执行剩下的异步任务

异步任务

异步任务分为微任务和宏任务,异步任务在执行的过程中会先执行微任务,再执行宏任务

常见的微任务和宏任务:

微任务:promise.then()、process.nextTick()、MutationObserver()

宏任务:setTimeout()、script标签、setInterval()、ajax、I/O、UI渲染

需要注意的是,所有的setTimeout()函数都共用同一个计时器,所有setTimeout()同时进行倒数,而不是代码执行到此才开始倒计时,所以计时短的会先出队列被执行

js 复制代码
setTimeout(() => {
  console.log('a')
}, 2000)

setTimeout(() => {
  console.log('b')
}, 1000)//b a

本该根据宏任务队列的先进先出顺序,会先输出a,再输出b,但是因为setTimeout()的这一特殊性造成这种结果

事件循环的机制:

所以事件循环机制是:

  1. v8执行时,从上往下执行,遇到异步函数就放入对应队列
  2. 去到微任务队列中按先进先出的顺序执行微任务
  3. 如果有需要,就渲染页面
  4. 去到宏任务队列中按先进先出的顺序执行微任务
  5. 当以上被执行完,说明这一次的循环结束,可以开启下一次循环了

asnyc/await 函数

相当于.then()的另一种写法

.then()的写法

js 复制代码
function A(){
return new Promise((resolve) => {
    setTimeout(() => {
        console.log('a');
        resolve()
    }, 1000)
})
}

function B(){
    console.log('b');
}

A().then(() => {
    B()
})

async/await的写法:

js 复制代码
function A(){
return new Promise((resolve) => {
    setTimeout(() => {
        console.log('a');
        resolve()
    }, 1000)
})
}

function B(){
    console.log('b');
}

async function fn(){
    await A()
    B()
}
fn()

需要注意的是,两种写法都导致B().then()属性,所以B()是微任务A()是同步任务

相关推荐
wuhen_n1 小时前
RAG 核心:向量嵌入与本地向量数据库实战
前端·langchain·ai编程
kisshyshy2 小时前
告别 Node 噩梦?用 Bun + TypeScript 像写诗一样调用大模型
前端·typescript
wuhen_n2 小时前
RAG 关键环节:文本分块策略与最优参数配置
前端·langchain·ai编程
狂炫冰美式2 小时前
AI 生成 Draw.io,导入飞书/Lark 画板后可编辑
前端·人工智能·后端
Moment2 小时前
我做了一套前端也能学懂的 AI Agent 系列,从 Prompt 一路讲到多 Agent 😍😍😍
前端·后端·面试
dy17172 小时前
二维码打印
前端·javascript·vue.js
智商不够_熬夜来凑2 小时前
【Radio & Checkbox】
前端·javascript·vue.js
xiaofeichaichai2 小时前
Diff 算法
前端·javascript
wgc2k3 小时前
Nest.js 基础-8-Hello,NestJS
开发语言·javascript·ecmascript