返回顶部
首页 > 资讯 > 前端开发 > JavaScript >JavaScript的三座大山之单线程和异步
  • 222
分享到

JavaScript的三座大山之单线程和异步

2024-04-02 19:04:59 222人浏览 泡泡鱼
摘要

目录一、进程与线程1. 进程:2. 线程:3. 进程与线程的关系:4. 浏览器运行是单进程还是多进程?5. 如何查看浏览器是否是多进程运行的呢?6. 浏览器运行是单线程还是多线程?二

一、进程与线程

1. 进程:

程序的一次执行, 它占有一片独有的内存空间  ---- 可以通过windows任务管理器查看进程;

2. 线程:

是进程内的一个独立执行单元;是程序执行的一个完整流程;CPU的基本调度单位;

3. 进程与线程的关系:

  * 一个进程中一般至少有一个运行的线程: 主线程 -- 进程启动后自动创建;

  * 一个进程中也可以同时运行多个线程, 我们会说程序是多线程运行的;

  * 一个进程内的数据可以供其中的多个线程直接共享;

  * 多个进程之间的数据是不能直接共享的

4. 浏览器运行是单进程还是多进程?

  • 有的是单进程
  • firefox
  • 有的是多进程
  • chrome

5. 如何查看浏览器是否是多进程运行的呢?

任务管理器==>进程

6. 浏览器运行是单线程还是多线程?

都是多线程运行的

二、单线程 

1、什么是单线程

JavaScript语言的一大特点就是单线程,也就是说同一时间只能做一件事。

//栗子? console.log(1)console.log(2)console.log(3)//输出顺序 1 2 3

2. JavaScript为什么是单线程

  • 首先是历史原因,在创建 javascript 这门语言时,多进程、多线程的架构并不流行,硬件支持并不好。
  • 其次是因为多线程的复杂性,多线程操作需要加,编码的复杂性会增高。
  • 最后与它的用途有关,作为浏览器脚本语言,JavaScript的主要用途是与用户互动,以及操作DOM,如果同时操作 DOM ,在多线程不加锁的情况下,最终会导致 DOM 渲染的结果不可预期。

为了利用多核CPU的计算能力,HTML5提出WEB Worker标准,允许JavaScript脚本创建多个线程,但是子线程完全受主线程控制,且不得操作DOM。所以,这个新标准并没有改变JavaScript单线程的本质。 

三、同步与异步

1、JS的 同步任务/异步任务

同步任务: 

主线程上排队执行的任务,只有一个任务执行完毕,才能执行一个任务;

所有同步任务都在主线程上执行,形成一个执行栈(execution context stack)。

 异步任务:主线程外执行的任务;在主线程之外还存在一个“任务队列”(task queue),当异步任务执行完成后会以回调函数的方式放入任务队列中等待,等主线程空闲时,主线程就会去事件队列中取出等待的回调函数放入主线程中进行执行。这个过程反复执行就形成了js的事件循环机制(Event Loop)。

//栗子? 
// 同步
console.log(1)
// 异步
setTimeout(()=>{
    console.log(2)
},100)
// 同步
console.log(3)
//输出顺序 1 3 2

2、 JavaScript为什么需要异步

如果在JS代码执行过程中,某段代码执行过久,后面的代码迟迟不能执行,产生阻塞(即卡死),会影响用户体验。

3、JavaScript怎么实现异步 

1)执行栈与任务队列 

其实上面我们已经提到了,JS实现异步时通过 事件循环

我们先理解几个概念:

  • JS任务 分为同步任务(synchronous)和异步任务(asynchronous)
  • 同步任务都在 JS引擎线程(主线程) 上执行,形成一个执行栈(call stack)
  • 事件触发线程 管理一个 任务队列(Task Queue)
  • 异步任务 触发条件达成,将 回调事件 放到任务队列(Task Queue)中
  • 执行栈中所有同步任务执行完毕,此时JS引擎线程空闲,系统会读取任务队列,将可运行的异步任务回调事件添加到执行栈中,开始执行

 当一个JS文件第一次执行的时候,js引擎会 解析这段代码,并将其中的同步代码 按照执行顺序加入执行栈中,然后从头开始执行。如果当前执行的是一个方法,那么js会向执行栈中添加这个方法的执行环境,然后进入这个执行环境继续执行其中的代码。当这个执行环境中的代码 执行完毕并返回结果后,js会退出这个执行环境并把这个执行环境销毁,回到上一个方法的执行环境。这个过程反复进行,直到执行栈中的代码全部执行完毕。

栗子 ?

//(1)
console.log(1)
//(2)
setTimeout(()=>{
    console.log(2)
},100)
//(3)
console.log(3)

1.先解析整段代码,按照顺序加入到执行栈中,从头开始执行

2.先执行(1),是同步的,所以直接打印 1

3.执行(2),发现是 setTimeout,于是调用浏览器的方法(webapi)执行,在 100ms后将 console.log(2) 加入到任务队列

4.执行(3),同步的,直接打印 3

5.执行栈已经清空了,现在检查任务队列,(执行太快的话可能此时任务队列还是空的,没到100ms,还没有将(2)的打印加到任务队列,于是不停的检测,直到队列中有任务),发现有 console.log(2),于是添加到执行栈,执行console.log(2),同步代码,直接打印 2 (如果这里是异步任务,同样会再走一遍循环:-->任务队列->执行栈)

所以结果是 1 3 2;

注意:setTimeout/Promise等我们称之为任务源。而进入任务队列的是他们指定的回调;

2)宏任务(macro task)与微任务(micro task)

 上面的循环只是一个宏观的表述,实际上异步任务之间也是有不同的,分为 宏任务(macro task) 与 微任务(micro task),最新的标准中,他们被称为 task与 jobs

  • 宏任务有哪些:script(整体代码), setTimeout, setInterval, setImmediate,I/O, UI rendering(渲染)
  • 微任务有哪些:process.nextTick, Promise, Object.observe(已废弃), MutationObserver(html5新特性)

下面我们再详细讲解一下执行过程:

 执行栈在执行的时候,会把宏任务放在一个宏任务的任务队列,把微任务放在一个微任务的任务队列,在当前执行栈为空的时候,主线程会 查看微任务队列是否有事件存在。如果微任务队列不存在,那么会去宏任务队列中 取出一个任务 加入当前执行栈;如果微任务队列存在,则会依次执行微任务队列中的所有任务,直到微任务队列为空(同样,是吧队列中的事件加到执行栈执行),然后去宏任务队列中取出最前面的一个事件加入当前执行栈...如此反复,进入循环。

注意:

  • 宏任务和微任务的任务队列都可以有多个
  • 当前执行栈执行完毕时会立刻先处理所有微任务队列中的事件,然后再去宏任务队列中取出一个事件。同一次事件循环中,微任务永远在宏任务之前执行。
  • 不同的运行环境 循环策略可能有不同,这里探讨chrome、node环境

 栗子 ? 

//(1)
setTimeout(()=>{
    console.log(1)   // 宏任务
},100)
//(2)
setTimeout(()=>{
    console.log(2)  // 宏任务
},100)
//(3)
new Promise(function(resolve,reject){
    //(4)
    console.log(3)  // 直接打印
    resolve(4)
}).then(function(val){
    //(5)
    console.log(val); // 微任务
})
//(6)
new Promise(function(resolve,reject){
    //(7)
    console.log(5)   // 直接打印
    resolve(6)
}).then(function(val){
    //(8)
    console.log(val);  // 微任务
})
//(9)
console.log(7)  // 直接打印
//(10)
setTimeout(()=>{
    console.log(8) // 宏任务,单比(1)(2)宏任务早
},50)

 上面的代码在node和chrome环境的正确打印顺序是 3 5 7 4 6 8 1 2

下面分析一下执行过程:

1.全部代码在解析后加入执行栈

2.执行(1),宏任务,调用webapi setTimeout,这个方法会在100ms后将回调函数放入宏任务的任务队列

3.执行(2),同(1),但是会比(1)稍后一点

4.执行(3),同步执行new Promise,然后执行(4),直接打印 3 ,然后resolve(4),然后.then(),把(5)放入微任务的任务队列

5.执行(6),同上,先打印 5 ,再执行resolve(6),然后.then()里面的内容(8)加入到微任务的任务队列

6.执行(9),同步代码,直接打印 7

7.执行(10),同(1)和(2),只是时间更短,会在 50ms 后将回调 console.log(8) 加入宏任务的任务队列

8.现在执行栈清空了,开始检查微任务队列,发现(5),加入到执行栈执行,是同步代码,直接打印 4

9.任务队列又执行完了,又检查微任务队列,发现(8),打印 6

10.任务队列又执行完了,检查微任务队列,没有任务,再检查宏任务队列,此时如果超过了50ms的话,会发现 console.log(8) 在宏任务队列中,于是执行 打印 8

11.依次打印 1 2

:因为渲染也是宏任务,需要在一次执行栈执行完后才会执行渲染,所以如果执行栈中同时有几个同步的改变同一个样式的代码,在渲染时只会渲染最后一个。

总结

本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注编程网的更多内容!      

--结束END--

本文标题: JavaScript的三座大山之单线程和异步

本文链接: https://lsjlt.com/news/163761.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • JavaScript的三座大山之单线程和异步
    目录一、进程与线程1. 进程:2. 线程:3. 进程与线程的关系:4. 浏览器运行是单进程还是多进程5. 如何查看浏览器是否是多进程运行的呢6. 浏览器运行是单线程还是多线程二、单线...
    99+
    2024-04-02
  • 浅谈JS三座大山之异步和单线程
    目录单线程异步单线程 但是我们在开发中,遇到请求网络,或者定时任务的时候,如果等待网络请求结束或者定时任务结束的时候再去做其他事情,这样页面就会卡住,所以js有异步机制解决这个问题。...
    99+
    2024-04-02
  • JavaScript的单线程和异步详细
    目录一、任务队列二、借以解释几个容易困惑的问题1、setTimeout(f1,0)是什么鬼2、Ajax请求是否异步3、界面渲染线程是单独开辟的线程三、如何利用浏览器的异步机制四、异步...
    99+
    2024-04-02
  • JavaScript单线程和异步怎么实现
    这篇“JavaScript单线程和异步怎么实现”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇...
    99+
    2024-04-02
  • JavaScript中的单线程和异步该如何理解
    小编今天带大家了解JavaScript中的单线程和异步该如何理解,文中知识点介绍的非常详细。觉得有帮助的朋友可以跟着小编一起浏览文章的内容,希望能够帮助更多想解决这个问题的朋友找到问题的答案,下面跟着小编一起深入学习“JavaScript中...
    99+
    2023-06-29
  • JS中异步和单线程的示例分析
    这篇文章主要介绍了JS中异步和单线程的示例分析,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。单线程但是我们在开发中,遇到请求网络,或者定时任务的时候,如果等待网络请求结束或者...
    99+
    2023-06-15
  • Flutter 异步编程之单线程下异步模型图文示例详解
    目录一、 本专栏图示概念规范1. 任务概念规范2. 任务的状态3. 时刻与时间线4.同步与异步二、理解单线程中的异步任务1. 任务的分配2.异步任务特点3. 异步任务完成与回调三、 ...
    99+
    2024-04-02
  • java 线程之对象的同步和异步(实例讲解)
    一、多线程环境下的同步与异步同步:A线程要请求某个资源,但是此资源正在被B线程使用中,因为同步机制存在,A线程请求不到,怎么办,A线程只能等待下去。package com.jalja.org.thread.demo01;public cla...
    99+
    2023-05-31
    java 线程 同步
  • PHP 多线程和异步编程的差异?
    php 多线程和异步编程的关键差异:多线程创建独立运行的线程,共享内存,但上下文切换成本高,且需要第三方扩展支持。异步编程使用事件循环处理并发请求,回调函数在事件循环中执行,php 内置...
    99+
    2024-05-06
    php 多线程 并发请求
  • 多线程的同步和异步学习
    1.创建多线程的方式有两种方式创建多线程,一种是继承Thread类,一种是实现Runnable接口;一个类如果继承了Thread类,同时覆写了本类中的run()方法进行启动,就可以实现多线程操作了.但是一个类只能继承一个父类.<1&g...
    99+
    2023-06-02
  • JavaScript异步编程之Promise的初步使用详解
    1. 概述 Promise对象是ES6提出的的异步编程的规范。说到异步编程,就不得不说说同步和异步这两个概念。 从字面意思理解同步编程的话,似乎指的是两个任务同步运行,如果这样理解就...
    99+
    2024-04-02
  • 异步编程:Python、Unix、JavaScript之间的差异和共同点是什么?
    异步编程是一种编程模型,它可以在程序执行其他任务的同时,处理多个并发的异步操作。Python、Unix和JavaScript都支持异步编程,但它们之间存在着差异和共同点。本文将探讨Python、Unix和JavaScript之间异步编程的差...
    99+
    2023-06-18
    unix javascript 异步编程
  • 异步编程:Python 和 javascript 之间的响应差距如何?
    异步编程是现代编程中越来越重要的一个概念,它能够提高程序的性能和效率。在这篇文章中,我们将讨论 Python 和 JavaScript 之间的异步编程响应差距,以及如何使用异步编程来提高程序的效率。 Python 和 JavaScript ...
    99+
    2023-07-27
    异步编程 响应 javascript
  • NPM、Python 和异步编程:这三者之间有何关联?
    在软件开发领域,NPM和Python都是非常流行的工具和编程语言。而异步编程则是一种重要的编程技术,它可以帮助开发人员更高效地利用计算机资源,提高程序性能。那么,这三者之间有何关联呢?本文将为您深入解答。 一、NPM和异步编程 NPM是一...
    99+
    2023-07-22
    numy 异步编程 npm
  • Javascript异步编程之你真的懂Promise吗
    目录前言基本用法语法错误处理Promise链式调用async & await常用的方法1、Promise.resolve()2、Promise.reject()3、Promi...
    99+
    2024-04-02
  • Laravel和JavaScript的异步编程与Python的异步编程有何不同?
    随着计算机技术的不断发展,异步编程已经成为了现代编程中的一个非常重要的概念。许多编程语言都提供了异步编程的支持,如Laravel、JavaScript和Python等。虽然它们都可以实现异步编程,但是它们之间的实现方式有所不同。本文将探讨...
    99+
    2023-09-08
    异步编程 laravel javascript
  • JavaScript三大重点同步异步与作用域和闭包及原型和原型链详解
    目录1. 同步、异步2. 作用域、闭包闭包作用域3. 原型、原型链原型(prototype)原型链 如图所示,JS的三座大山: 同步、异步作用域、闭包原型、原型链 1. 同步、异步 ...
    99+
    2024-04-02
  • android开发教程之handle实现多线程和异步处理
    这次浅谈一下Handler,为什么会出现Handler这个功能特性呢?首先,在之前的基本控件,基本都是在Activity的onCreate(Bundle savedInstan...
    99+
    2022-06-06
    异步 android开发 多线程 线程 教程 Android
  • python并发编程之多进程、多线程、异步和协程详解
    最近学习python并发,于是对多进程、多线程、异步和协程做了个总结。 一、多线程 多线程就是允许一个进程内存在多个控制权,以便让多个函数同时处于激活状态,从而让多个函数的操作同时运行。即使是单CPU的计...
    99+
    2022-06-04
    之多 多线程 详解
  • PHP中的线程池和异步编程实践
    近年来,PHP 开发者们开发出许多并发编程的技术,其中又以线程池和异步编程应用最为广泛。本文将介绍线程池和异步编程的基本概念以及它们在 PHP 中的实践应用。 一、线程池的概念线程池是一种并发编程领域中常见的技术。它是由一组线程组成的线程集...
    99+
    2023-05-23
    线程池 异步编程 PHP
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作