Agent:
-
共享一个JavaScript对象的页面集合,如
<iframe>
或window.open
(相似来源的窗口代理) -
Worker
和Worklet
属于与主页面不同的Agent(Dedicated worker agent
,Shared worker agent
,Service worker agent
,Worklet agent
) -
因为
Agent
共享一个事件循环,所以在Agent
中总是最多只有一段JavaScript代码在运行
Agent Cluster:
-
SharedArrayBuffer
是一个共享内存的Agent集合 -
当页面本身的JavaScript和Web Worker中的JavaScript共享一个
SharedArrayBuffer
时,它就成为一个代理集群
除Agent Cluster之外的:
- 通过消息传递的交互,如
postMessage
,没有特别的定义,因为它们不需要被建模为JavaScript规范
在Agent外部,可以使用线程并行,但在Agent内部,异步处理是使用任务并行,而不是线程并行。在下文中,我们将重点讨论Agent内部的异步处理问题(尤其是在同一领域内)
======================================================================
事实上,JavaScript没有并行性(在每个代理的基础上),这是一个强大的功能,使大多数处理在本质上是原子性的
// 不包括 await 或 then。这是在原子状态下执行的
this.counter++;
这就是为什么像Mutex这样熟悉的模式不会像其他语言那样经常出现在JavaScript中。
另一方面,这种约束可能会阻碍并行性和并发性的有效实施
=======================================================================
在JavaScript中,事件循环是由语言处理器管理的,而应用程序的执行是服从于它的。从事件循环中执行的单段JavaScript代码在ECMAScript中被称为"作业(Host)"
window.onload = () => { console.log(“loaded”); }
// ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ 负载完成后,这将被添加到作业队列中。
setTimeout(() => console.log(“1s has passed”), 1000);
// ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ 这将在1秒后被添加到作业队列中。
一次只能有一个工作在运行。当一个作业完成后(例如最外层的函数返回),下一个作业被执行。
工作没有统一的优先级,可能会被之前排队的工作打断。
在调度方面,主机环境并不要求统一对待Job。例如,网络浏览器和Node.js对待Promis-handling例如,网络浏览器和Node.js将处理承诺的工作视为比其他工作更优先;未来的功能可能会增加不以如此高优先级处理的
工作
。
然而,规定与 "承诺 "有关的工作必须在同一个队列中。
工作 必须按照安排它们的HostEnqueuePromiseJob调用的相同顺序运行。
这确保了以下程序将依次输出0-9。
(async () => { for(let i = 0; i < 10; i += 2) console.log(await i); })()。
(async () => { for(let i = 1; i < 10; i += 2) console.log(await i); })()。
截至2021年,ECMAScript标准中没有定义与Promise无关的工作。这些将在其他规范中被额外定义,如HTML。
从今以后,我们将使用自己的术语,把一个microtick定义为Promise作业队列的一个周期。
=============================================================================
ECMAScript作业在HTML规范中被称为任务。队列中还加载了ECMAScript作业以外的任务。该队列分为两种类型
-
正常)任务队列
-
可以有多个(正常)任务队列,用于不同类型的任务(取决于实施)。
-
微任务队列
-
排在微任务队列中的任务被称为微任务
在ECMAScript规范中,所有的任务队列和微任务队列一起被认为是一个作业队列
requestIdleCallback中定义的空闲回调列表/可运行回调列表也可以被视为任务队列的一个变体
事件循环的处理在§8.1.6.3中规定,与本文相关的内容可归纳为以下几点
-
如果在任务队列中有一个任务,它将被检索和执行。
-
微任务队列中的任务被获取并逐一执行,直到没有更多的任务。
-
更新图纸。
-
如果任务队列中有空间,背景任务将根据 RequestIdleCallback 规范进行排队。
-
从头开始
这意味着
-
微型任务比普通任务有优先权。
-
微型任务的优先级高于绘图。
微任务比普通任务有优先权,这意味着它们会饿死。下面的等待忙碌循环对浏览器的负面影响几乎和无限循环一样,因为它阻断了绘图。
(async () => {
while(true) {
await null;
}
})();
有两种典型的方法来排队等候微任务
-
功能
排队微任务
-
Promise.prototype.then函数。这个实体的HTML规范实现, HostEnqueuePromiseJob,要被加载到一个微任务队列中。
-
这也是async/await中的await的情况。
另一方面,要显式地排队等候一个正常的任务并不容易。另一方面,要显式地对一个普通的任务进行排队并不容易,因为任务通常是由一些事件来排队的。 在 用于网络浏览器环境的setImmediate的polyfill中,使用了以下实现。
-
在窗口代理下,你可以通过向自己发送 window.postMessage 来启动一个事件
-
在Worker代理下,使用 MessageChannel postMessage
-
使用 setTimeout 作为退路
我们使用自己的术语,将一个microtick定义为网络浏览器环境中微任务队列的一个周期。这与上一节中的定义是一致的。我们还将 任务队列周期定义 为一个tick,这是我们自己的术语。
==============================================================================
Node.js是基于Chrome的JavaScript引擎V8,任务的概念与网络浏览器相似。(queueMicrotask 和 Promise.prototype.then 是可用的)
除了微任务队列之外 process.nextTick 队列是Node.js的一个特性。一般来说, process.nextTick 优先于微任务,但如果它在一个微任务中被排队,它的处理优先级将低于其他微任务。这是因为任务队列是按以下顺序处理的
while(true) {
waitForAnyTask();
if (taskQueue.length > 0) taskQueue.pop().run();
do {
while (nextTickQueue.length > 0) nextTickQueue.pop().run();
while (microtaskQueue.length > 0) microtaskQueue.pop().run();
} while (nextTickQueue.length > 0);
}
process.nextTick 的存在是为了Node.js内置的I/O处理和历史原因。对于Node.js中内置的I/O处理,由于历史原因,现在推荐使用 queueMicrotask 。
与浏览器中的JavaScript不同,Node.js还提供了一个函数 setImmediate ,可用于将任务排入常规任务队列 。它的使用方式与 setTimeout 基本相同。
// 在执行之前处理当前的I/O任务等。
setImmediate(() => console.log(“Hello!”))
在Node.js环境中,微任务队列的一个周期在我们自己的术语中被定义为一个Microtick。这与上一节中的定义是一致的。我们也把我们自己的术语定义 为一个任务队列周期,即一个tick
=============================================================================
一个旧的API用于在一定时间后排队任务,而不是直接排队,这就是 setTimeout ,它在网络浏览器和Node.js中都有实现。
setTimeout(() => console.log(“1 second has passed”), 1000);
网络浏览器和Node.js之间有许多不同之处
-
最小秒数的行为差异(见下文)。
-
在Node.js中,处理程序不能作为字符串传递。
-
在Node.js中,返回的是一个定时器对象(Timeout),而不是一个定时器ID(数字)。
-
Node.js允许对定时器事件进行ref/unref操作。
最后
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。
深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长,自己不成体系的自学效果低效漫长且无助。
因此收集整理了一份《2024年Web前端开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点!不论你是刚入门Android开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门!
如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!
由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!
img-CNEL4VWC-1715807731656)]
[外链图片转存中…(img-BEr0zotV-1715807731656)]
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点!不论你是刚入门Android开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门!
如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!
由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!