@charset "UTF-8";
.markdown-body { line-height: 1.75; font-weight: 400; font-size: 15px; overflow-x: hidden; color: rgba(43, 43, 43, 1); font-family: -apple-system, system-ui, BlinkMacSystemFont, Helvetica Neue, PingFang SC, Hiragino Sans GB, Microsoft YaHei, Arial, sans-serif; background-image: linear-gradient(90deg, rgba(159, 219, 252, 0.15) 3%, rgba(0, 0, 0, 0) 0), linear-gradient(1turn, rgba(159, 219, 252, 0.15) 3%, rgba(0, 0, 0, 0) 0); background-size: 20px 20px; background-position: center }
.markdown-body h1, .markdown-body h2, .markdown-body h3, .markdown-body h4, .markdown-body h5, .markdown-body h6 { padding: 30px 0; margin-top: 35px; margin-bottom: 10px; color: rgba(77, 208, 225, 1) }
.markdown-body h1 { font-size: 30px; text-align: center; position: relative; width: max-content; margin: 0 auto }
.markdown-body h1:before { position: absolute; content: ""; z-index: -1; top: -20px; height: 100%; width: 100px; left: 0; right: 0; margin: 0 auto; background: url("") center / 64px 64px no-repeat; opacity: 0.84 }
.markdown-body h1:after { position: absolute; content: ""; width: 150%; left: -25%; height: 50%; bottom: 12px; border-radius: 50%; background: linear-gradient(rgba(0, 0, 0, 0) 80%, rgba(77, 208, 225, 0.8)); opacity: 0.6; animation: 6s linear infinite h1animate }
@keyframes h1Animate { 0% { background-position: right bottom } 50% { background-position: right } 100% { background-position: right bottom } }
.markdown-body h2 { display: block; border-bottom: 4px solid rgba(77, 208, 225, 1); position: relative; font-size: 24px; padding: 12px 32px; margin: 30px 0 }
.markdown-body h2:before { width: 24px; height: 24px; left: 0; top: 0; margin: auto; background-size: 24px 24px; background-image: url("") }
.markdown-body h2:after, .markdown-body h2:before { content: ""; display: block; position: absolute; bottom: 0 }
.markdown-body h2:after { right: 0; width: 400px; height: 10px; border-top-right-radius: 24px; background: linear-gradient(90deg, rgba(255, 255, 255, 1), rgba(77, 208, 225, 1)); max-width: 50vw }
.markdown-body h3 { margin: 30px 0; font-size: 18px; position: relative; padding: 4px 32px; width: max-content }
.markdown-body h3:before { border-bottom: 2px solid rgba(77, 208, 225, 1); width: 100%; content: ""; display: block; height: 28px; position: absolute; left: 0; top: 0; bottom: -2px; margin: auto; background-size: 28px 28px; background-image: url(""); background-repeat: no-repeat; animation: 2s infinite alternate h3animationbefore }
@keyframes h3AnimationBefore { 0% { width: 28px } 25% { width: 100% } 50% { width: 100% } 100% { width: 100% } }
.markdown-body h3:after { content: ""; display: block; width: 28px; height: 28px; position: absolute; border: 2px solid rgba(77, 208, 225, 1); border-radius: 50%; right: -15px; top: 0; bottom: 0; margin: auto; background-size: 28px 28px; background-image: url(""); animation: 2s infinite alternate h3animationafter }
@keyframes h3AnimationAfter { 0% { } 10% { } 50% { transform: rotate(-1turn) } 100% { transform: rotate(-1turn) } }
.markdown-body h4 { font-size: 16px }
.markdown-body h5 { font-size: 15px }
.markdown-body h6 { margin-top: 5px }
.markdown-body p { line-height: inherit; margin: 22px 0; letter-spacing: 2px; font-size: 14px; word-spacing: 2px }
.markdown-body img { max-width: 80%; border-radius: 6px; display: block; margin: 20px auto !important; object-fit: contain; box-shadow: 0 0 16px rgba(110, 110, 110, 0.45) }
.markdown-body figcaption { display: block; font-size: 13px; color: rgba(43, 43, 43, 1) }
.markdown-body figcaption:before { content: ""; background-image: url(""); display: inline-block; width: 18px; height: 18px; background-size: 18px; background-repeat: no-repeat; background-position: center; margin-right: 5px; margin-bottom: -5px }
.markdown-body hr { border-top: 1px solid rgba(77, 208, 225, 1); border-right: none; border-bottom: none; border-left: none; margin-top: 32px; margin-bottom: 32px }
.markdown-body del { color: rgba(77, 208, 225, 1) }
.markdown-body code { border-radius: 2px; overflow-x: auto; background-color: rgba(77, 208, 225, 0.08); color: rgba(38, 198, 218, 1); padding: 0.195em 0.4em }
.markdown-body pre { font-family: Menlo, Monaco, Consolas, Courier New, monospace; overflow: auto; position: relative; line-height: 1.75; box-shadow: 0 0 8px rgba(110, 110, 110, 0.45); border-radius: 4px; margin: 16px }
.markdown-body pre:before { content: ""; display: block; height: 30px; width: 100%; margin-bottom: -7px; background: url("") 10px 10px / 40px no-repeat }
.markdown-body pre>code { font-size: 12px; padding: 15px 12px; margin: 0; word-break: normal; display: block; overflow-x: auto; color: rgba(51, 51, 51, 1); background: rgba(248, 248, 248, 1) }
.markdown-body a { color: rgba(77, 208, 225, 1); border-bottom: 1px solid rgba(77, 208, 225, 1); font-weight: 400; text-decoration: none; margin: 0 4px }
.markdown-body a:active, .markdown-body a:hover { background-color: rgba(77, 208, 225, 0.1) }
.markdown-body strong { color: rgba(38, 198, 218, 1) }
.markdown-body strong:before { content: "「" }
.markdown-body strong:after { content: "」" }
.markdown-body em { font-style: normal; color: rgba(77, 208, 225, 1); font-weight: 700 }
.markdown-body table { display: inline-block !important; font-size: 12px; width: auto; max-width: 100%; overflow: auto; border: 1px solid rgba(246, 246, 246, 1) }
.markdown-body thead { background: rgba(246, 246, 246, 1); color: rgba(0, 0, 0, 1); text-align: left }
.markdown-body tr:nth-child(2n) { background-color: rgba(77, 208, 225, 0.05) }
.markdown-body td, .markdown-body th { padding: 12px 7px; line-height: 24px }
.markdown-body td { min-width: 120px }
.markdown-body blockquote { margin: 2em 0; padding: 24px 32px; border-left: 4px solid rgba(38, 198, 218, 1); background: rgba(77, 208, 225, 0.15); position: relative }
.markdown-body blockquote:before { content: "❝"; top: 8px; left: 8px; color: rgba(77, 208, 225, 1); font-size: 30px; line-height: 1; font-weight: 700; position: absolute; opacity: 0.7 }
.markdown-body blockquote:after { content: "❞"; font-size: 30px; position: absolute; right: 8px; bottom: 0; color: rgba(77, 208, 225, 1); opacity: 0.7 }
.markdown-body blockquote p { color: rgba(89, 89, 89, 1); line-height: 2 }
.markdown-body ol, .markdown-body ul { color: rgba(89, 89, 89, 1); padding-left: 28px }
.markdown-body ol li, .markdown-body ul li { margin-bottom: 0; list-style: inherit }
.markdown-body ol li .task-list-item, .markdown-body ul li .task-list-item { list-style: none }
.markdown-body ol li .task-list-item ol, .markdown-body ol li .task-list-item ul, .markdown-body ul li .task-list-item ol, .markdown-body ul li .task-list-item ul { margin-top: 0 }
.markdown-body ol ol, .markdown-body ol ul, .markdown-body ul ol, .markdown-body ul ul { margin-top: 3px }
.markdown-body ol li { padding-left: 6px }
@media (max-width: 720px) { .markdown-body h1 { font-size: 24px } .markdown-body h2 { font-size: 20px } .markdown-body h3 { font-size: 18px } }.markdown-body pre, .markdown-body pre>code.hljs { background: rgba(255, 255, 255, 1); color: rgba(0, 0, 0, 1) }
.hljs-comment, .hljs-quote, .hljs-variable { color: rgba(0, 128, 0, 1) }
.hljs-built_in, .hljs-keyword, .hljs-name, .hljs-selector-tag, .hljs-tag { color: rgba(0, 0, 255, 1) }
.hljs-addition, .hljs-attribute, .hljs-literal, .hljs-section, .hljs-string, .hljs-template-tag, .hljs-template-variable, .hljs-title, .hljs-type { color: rgba(163, 21, 21, 1) }
.hljs-deletion, .hljs-meta, .hljs-selector-attr, .hljs-selector-pseudo { color: rgba(43, 145, 175, 1) }
.hljs-doctag { color: rgba(128, 128, 128, 1) }
.hljs-attr { color: rgba(255, 0, 0, 1) }
.hljs-bullet, .hljs-link, .hljs-symbol { color: rgba(0, 176, 232, 1) }
.hljs-emphasis { font-style: italic }
.hljs-strong { font-weight: 700 }

日常刷JS面试题,难免会遇到这么一道:请简述JS的事件驱动模型。各位小伙伴是否也曾经有过这样的经历,对着这道题,一脸懵逼,一脸蒙圈。今天,掌门人就来给大家普及一下 JS 事件驱动模型的相关知识。

JS异步编程的背景:从回调地狱到Promise和async/await

在了解事件驱动模型之前,我们先回顾一下JS异步编程的历史。

记得刚开始写JS的时候,还没有现在流行的Promise和async/await,那时候我们可能还在回调地狱里挣扎。比如,要实现服务端的请求数据一般需要这样写:

getDataFromServer(params, function(err, result1){
if (!err) {
getNextResult(result1, function(err, result2) {
if (!err) {
getFinalResult(result2, function(err, result3) {
if (!err) {
// Do something with the result
} else {
console.log(err);
}
}
} else {
console.log(err);
}
});
} else {
console.log(err);
}
});

这样的代码对于我们来说,一股绝望,如果嵌套多了,直接影响开发者的心情。但随着时间的推移,Promise和async/await逐渐逐渐变得越来越流行,如今已成为JS异步编程的标配。

用Promise和async/await实现上面代码的异步请求,大家一定更为熟悉,更为好看:

async function getDataFromServer(params) {
try {
const result1 = await fetch(url1);
const result2 = await fetch(url2 + result1.id);
const finalResult = await fetch(url3 + result2.id);
// Do something with finalResult
} catch (err) {
console.log(err);
}
}

异步编程变得更加方便了,但是背后却有什么本质的变化吗?实质上,JS的异步编程背后采用了事件驱动模型。

什么是JS事件驱动模型:理解事件循环、宏任务与微任务

事件驱动模型是什么?简单说来,是当特定事件发生时,某个函数或代码块会被触发和执行的一种编程方式。

JS事件驱动模型的核心是事件循环。所谓的事件循环就是一个不断循环的过程,JS引擎会检查宏队列和微队列里的任务有没有被处理完,如果宏队列和微队列均为空,则JS引擎会一直休眠,等待宏队列和微队列中被新加入的任务唤醒。其中, 宏任务存放在宏任务队列中,常见的宏任务有setTimeout、setInterval、setImmediate和I/O操作等。而微任务则处于微任务队列中,常见的微任务有Promise.then、process.nextTick和object.observe,主要用于对象数据的监听。

console.log('script start');

setTimeout(function() {
console.log('setTimeout');
}, 0); Promise.resolve().then(function() {
console.log('promise1');
}).then(function() {
console.log('promise2');
}); console.log('script end');

运行上述代码,你会发现输出顺序是这样的:

script start
script end
promise1
promise2
setTimeout

我们来看一下这段代码的执行过程:

  • 执行第一行代码,输出 "script start"。
  • 设置一个0毫秒后执行的定时器,这个定时器是一个宏任务,被放到宏任务队列中。
  • 解析Promise.resolve()语句,并将其添加到微任务队列中。因为微任务队列是在当前任务完成后立即执行的,所以接下来会立即执行微任务。
  • 执行第7行代码,输出 "script end"。
  • 当前任务同步代码执行完成,开始执行微任务。
  • 执行第10行代码,输出 "promise1",继续执行第11行代码,输出 "promise2"。
  • 微任务队列为空,开始执行宏任务队列中的任务。
  • 执行定时器回调函数,输出 "setTimeout"。

可以看到,微任务的执行优先于宏任务,而不同类型的任务又是按照队列的顺序依次执行的。了解这些背后的机制,会帮助我们更好地理解事件驱动模型在JS中的应用。

浏览器中的事件机制:从事件触发到事件处理的整个流程

在浏览器中,事件是如何被触发和处理的呢?下面是整个流程的简述:

  1. 用户在浏览器中触发一个事件,比如点击鼠标、按下键盘等。

  2. 事件被封装成Event对象。

  3. Event对象被放入宏任务队列中,等待JS引擎执行。

  4. 当宏任务队列中的事件被执行时,JS引擎会把这个Event对象拿出来,并调用该事件对应的监听器或回调函数。

  5. 监听器或回调函数执行完成后,如果有相关的异步请求,JS引擎会把它们放入微任务队列中,等待当前宏任务执行完成后执行。

document.addEventListener('click', function(event) {
console.log('You clicked the button');
});

在上述代码中,当用户点击页面中的某个元素时,就会触发click事件并被封装为Event对象,然后被添加到宏任务队列中等待执行。当JS引擎执行到这个宏任务时,会调用监听器函数并输出 "You clicked the button"。

常见事件类型的处理方式:鼠标/键盘事件、网络请求、DOM操作等

在浏览器中,事件类型很多,处理方式也各不相同。常见的事件类型包括鼠标/键盘事件、页面加载事件、网络请求事件、DOM操作等。下面是一些常见事件类型和处理方式的示例。

鼠标/键盘事件

document.addEventListener('mousemove', function(event) {
console.log('Mouse position:', event.clientX, event.clientY);
}); document.addEventListener('keydown', function(event) {
console.log('Key pressed:', event.keyCode);
});

当用户移动鼠标时,mousemove事件会被触发并产生一个Event对象,其中clientX和clientY属性表示鼠标的坐标位置。当用户按下键盘时,keydown事件也会产生一个Event对象,其中keyCode表示按下的键位。

网络请求事件

const req = new XMLHttpRequest();
req.open('GET', 'https://example.com/data');
req.addEventListener('load', function(event) {
console.log('Data loaded:', req.responseText);
});
req.send();

使用XMLHttpRequest发送HTTP请求,当服务器返回响应时,load事件将被触发并产生一个Event对象,其中可以通过responseText属性获取服务器响应的数据。

DOM操作

const btn = document.getElementById('my-button');
btn.addEventListener('click', function(event) {
event.preventDefault();
console.log('Button clicked');
// do something else
});

当用户点击按钮时,click事件会被触发并产生一个Event对象,其中preventDefault方法可以阻止默认行为,比如链接跳转等。

JS事件驱动模型的优化和性能问题:如何避免长任务影响用户体验?

在JS事件驱动模型的应用中,有一些常见的问题需要注意。其中一个重要的问题就是长任务的执行可能会导致UI线程阻塞,造成用户体验的不良影响。为了避免这种情况,我们需要优化代码,让长任务异步执行,并通过一些非阻塞的方式来处理。

代码示例:

const btn = document.getElementById('my-button');
btn.addEventListener('click', function(event) {
console.log('Button clicked');
setTimeout(function() {
console.log('Long task started');
// do something that takes a long time
console.log('Long task finished');
}, 0);
});

在上述代码中,当用户点击按钮时,会产生一个宏任务,其中的长任务被封装在一个定时器函数中并异步执行,从而不会阻塞UI线程的执行。

其他常见问题和应用场景:事件委托、事件冒泡、代码调试等

除了上述问题外,JS事件驱动模型还涉及到诸如事件委托、事件冒泡、代码调试等问题。下面我们来简要介绍这些问题和应用场景。

事件委托

事件委托指的是将事件绑定在父节点上,从而将事件的处理交给子节点。在大量的DOM节点绑定事件处理程序时,事件委托可以提高性能,避免内存泄漏等问题。

<ul id="my-list">
<li>Item 1</li>
<li>Item 2</li>
<li>Item 3</li>
<li>Item 4</li>
</ul> // 绑定事件委托
const list = document.getElementById('my-list');
list.addEventListener('click', function(event) {
const target = event.target;
if (target.tagName === 'LI') {
console.log('Clicked item:', target.textContent);
}
});

在上述代码中,我们将点击事件绑定在ul元素上,而不是每个li元素上,从而减少了事件处理程序的数量和内存使用。

事件冒泡

事件冒泡指的是当某个元素触发了事件时,它的父元素也会触发该事件。

<div id="parent">
<div id="child"></div>
</div> // 绑定事件冒泡
const parent = document.getElementById('parent');
parent.addEventListener('click', function(event) {
console.log('Parent clicked:', event.target.id);
}); const child = document.getElementById('child');
child.addEventListener('click', function(event) {
console.log('Child clicked:', event.target.id);
});

在上述代码中,当点击child元素时,parent元素会触发点击事件,由于事件冒泡的机制,父元素和子元素的事件处理程序均会被执行。

代码调试

JS事件驱动模型中涉及到的事件和任务可能是异步的,我们需要使用适当的调试技术来调试代码,以便更好地理解事件驱动模型的实现和调试代码。

结论和展望:JS事件驱动模型的未来发展趋势

在未来,JS事件驱动模型可能会更加智能化,一些新的API和机制也将会出现,以便更好地满足业务需求和提高开发效率。

此外,技术不断更新,在了解JS事件驱动模型的基础上,我们还需要不断学习新的技术知识,不断探索和实践,才能不断提升自己的技能水平。

当然,在使用JS事件驱动模型时还需要注意效率和可维护性等问题,不断优化代码,提高性能和可读性,从而更好地应对各种场景的需求。

最后,我希望通过这篇文章,能够帮助大家更好地理解JS事件驱动模型的基本概念、应用场景和相关技巧,从而更好地编写和优化JavaScript代码。

深度解析JS事件驱动模型:如何理解浏览器中的异步回调和事件循环的更多相关文章

  1. Node.js事件驱动模型

    一.传统线程网络模型 在了解Node.js事件驱动模型之前,我们先了解一下传统的线程网络模型,请求进入web服务器(IIS.Apache)之后,会在线程池中分配一个线程来线性同步完成请求处理,直到请求 ...

  2. 深入理解C#中的异步(一)——APM模式EAP模式

    深入理解C#中的异步(一)--APM模式EAP模式 目录 深入理解C#中的异步(一)--APM模式EAP模式 1 使用异步编程的原因 2 异步编程模式 2.1 APM模式 2.1.1 APM模式示例代 ...

  3. Java学习疑惑(8)----可视化编程, 对Java中事件驱动模型的理解

    我们编写程序就是为了方便用户使用, 我觉得UI设计的核心就是简洁, 操作过于繁琐的程序让很大一部分用户敬而远之. 即使功能强大, 但是人们更愿意使用易于操作的软件. 近年流行起来的操作手势和逐渐趋于成 ...

  4. spring事件驱动模型--观察者模式在spring中的应用

    spring中的事件驱动模型也叫作发布订阅模式,是观察者模式的一个典型的应用,关于观察者模式在之前的博文中总结过,http://www.cnblogs.com/fingerboy/p/5468994. ...

  5. node.js的作用、回调、同步异步代码、事件循环

    http://www.nodeclass.com/articles/39274 一.node.js的作用 I/O的意义,(I/O是输入/输出的简写,如:键盘敲入文本,输入,屏幕上看到文本显示输出.鼠标 ...

  6. 深入理解nodejs中的异步编程

    目录 简介 同步异步和阻塞非阻塞 javascript中的回调 回调函数的错误处理 回调地狱 ES6中的Promise 什么是Promise Promise的特点 Promise的优点 Promise ...

  7. [JS]异步任务之事件循环

    前言 常常会听到单线程和多线程这两个名词,单线程即一个时间段内程序从上到下执行任务,多线程即一个时间段内程序同时执行多个任务. 然而 JavaScript 是单线程的,它不像 Java 那样新开启一个 ...

  8. 通过作用域链解析js函数一些难以理解的的作用域问题

    基本原理 js函数在执行时,系统会创建一个隐式的属性scope,scope中存储的是函数的作用域链. 通过对这个scope的分析,就能解释JavaScript中许多难以理解的问题: 例1: funct ...

  9. spring5 源码深度解析----- 事务增强器(100%理解事务)

    上一篇文章我们讲解了事务的Advisor是如何注册进Spring容器的,也讲解了Spring是如何将有配置事务的类配置上事务的,实际上也就是用了AOP那一套,也讲解了Advisor,pointcut验 ...

  10. js-移动端android浏览器中input框被软键盘遮住的问题解决方案

    我遇到的问题:在一个页面里有一个弹出层之前我给我的最外层加了固定定位 用了下面的方法也不好使:没有办法我将之改为绝对定位层级变高在加上一个顶部标签通过js计算顶部高度来实现满屏遮挡: <sect ...

随机推荐

  1. Win10打开IE自动跳转至Edge解决办法

    WIN + R输入inetcpl.cpl弹出Internet属性对话窗口 点击上面菜单中的[高级]选项 滑动右侧滚动条,找到[浏览]项下面的[启用第三方浏览器拓展*]并取消勾选 双击IE浏览器图标测试 ...

  2. 08_使用python 内置 json 实现数据本地持久化

    使用python 内置 json 实现数据本地持久化 四个json函数 函数 json.load() 将本地json数据文件读取出来,并以列表形式返回从文件对象中读取 JSON 格式的字符串,并将其反 ...

  3. 【ABAQUS2023-Output Vars】使用记录

    计算结构的应变能,ALLSE=所有单元的ESEDEN*EVOL.但这不适用于模态分析,因为模态分析EVOL不能用 ALLSE Field: no History: yes .fil: automati ...

  4. 通过Kube-rbac-proxy保护 Kubernetes 工作负载中的应用容器

    1.概述 kube-rbac-proxy 是 Kubernetes 生态中一个专注于"基于角色的访问控制(RBAC)"的轻量级代理组件,通常以 Sidecar 容器的形式部署在 P ...

  5. HarmonyOS应用开发者高级认证【考题+答案】

    HarmonyOS应用开发者高级认证 前言 考试简介 掌握鸿蒙的核心概念和端云一体化开发.数据.网络.媒体.并发.分布式.多设备协同等关键技术能力,具备独立设计和开发鸿蒙应用能力. 博文说明 本博文的 ...

  6. go json omitempty 关键字 脱坑

    用法 大家对于 json 和 struct 之间的转换一定不陌生,为了将代码中的结构体与 json 数据解耦,通常我们会在结构体的 field 类型后加上解释说明,例如在表示一个地址的时候, json ...

  7. 项目管理协作工具对比:PingCode vs Leangoo

    多语言适配能力 在全球化协作场景下,多语言支持成为跨国团队的硬性指标.PingCode目前仅支持中文界面,对于涉及多国语言协作的团队存在使用局限.对比Leangoo提供中英文双语界面切换功能,可满足基 ...

  8. Unsloth更快训练大模型并导出GGUF - Windows

    环境搭建 系统环境 需要Nvidia显卡,至少8G显存,且专用显存与共享显存之和大于20G 建议将非安装版的环境文件都放到非系统盘,方便重装或移植 以Windows11为例,非安装环境文件都放在 E ...

  9. MySQL基础架构-架构详解

    mysql基础架构图 架构详解 连接器部分 整体功能 建立连接 维持管理连接 校验用户名密码,查询权限 最佳实践 不要在命令行客户端中明文输入密码 -p 中不要使用明文密码 修改权限,需要重新建立连接 ...

  10. 【SpringMVC】视图和视图解析器

    视图和视图解析器 Spring MVC如何解析视图 视图和视图解析器 请求处理方法执行完成后,最终返回一个 ModelAndView对象.对于那些返回 String,View 或 ModeMap 等类 ...