0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

JAVASCRIPT與單個線程的工作

汽車玩家 ? 來源:今日頭條 ? 作者:魏建民 ? 2020-05-05 22:07 ? 次閱讀

事件循環(huán)是用來理解JavaScript的最重要的方面之一。這篇文章旨在解釋JavaScript如何與單個線程一起工作的細節(jié),以及它如何處理異步函數(shù)。

JavaScript代碼運行是單線程。一次只執(zhí)行一件事。這實際上是一個非常有用的限制,因為它簡化了很多程序,從而不必擔心并發(fā)問題。

您只需要注意編寫代碼的方式,避免任何可能阻塞線程的內(nèi)容,如同步調(diào)用或無限循環(huán)。

通常,在大多數(shù)瀏覽器中,每個瀏覽器都有一個事件循環(huán),以使每個進程隔離,并避免web頁面具有無限循環(huán)或繁重的處理來阻塞整個瀏覽器。

你最需要擔心的是,您的代碼將在單個事件循環(huán)上運行,并在編寫代碼時考慮到這一點,以避免阻塞它。

阻止事件循環(huán)

任何花費太長時間將控制權(quán)返回給事件循環(huán)的JavaScript代碼都會阻止頁面中任何JavaScript代碼的執(zhí)行,甚至阻止UI線程,用戶也無法點擊,滾動頁面等等。

幾乎所有JavaScript中的I / O操作都是非阻塞的。網(wǎng)絡(luò)請求,Node.js文件系統(tǒng)操作等。阻塞是個例外,這就是為什么JavaScript基于回調(diào),以及最近的promises和async / await。

調(diào)用堆棧

調(diào)用堆棧是LIFO隊列(Last In,F(xiàn)irstOut)。事件循環(huán)不斷檢查調(diào)用堆棧以查看是否存在需要運行的任何函數(shù)。

在執(zhí)行此操作時,它會將它找到的任何函數(shù)調(diào)用添加到調(diào)用堆棧并按順序執(zhí)行每個調(diào)用。

一個簡單的事件循環(huán)說明:

JAVASCRIPT與單個線程的工作

當此代碼運行時,首先foo()調(diào)用。在foo()我們第一次調(diào)用bar(),然后我們調(diào)用baz()。

排隊功能執(zhí)行

上面的例子運行特點:JavaScript找到要執(zhí)行的東西,按順序運行它們。

讓我們看看如何推遲函數(shù)直到堆棧清除:

用例setTimeout(()=> {}), 0)是調(diào)用一個函數(shù),但是一旦執(zhí)行了代碼中的每個其他函數(shù)就執(zhí)行它。

JAVASCRIPT與單個線程的工作

當此代碼運行時,首先調(diào)用foo()。在foo()里面我們首先調(diào)用setTimeout,bar作為參數(shù)傳遞,然后我們指示它盡可能快地運行,將0作為計時器傳遞。然后我們調(diào)用baz()。

消息隊列

調(diào)用setTimeout()時,瀏覽器或Node.js啟動計時器。當計時器到期,我們將0作為超時,回調(diào)函數(shù)立即被放入消息隊列中。

消息隊列也是用戶發(fā)起的事件(如單擊事件、鍵盤事件或獲取響應(yīng))在代碼有機會對其作出響應(yīng)之前排隊的地方。或者像onLoad這樣的DOM事件。

循環(huán)優(yōu)先處理調(diào)用堆棧,它首先處理在調(diào)用堆棧中找到的所有東西,一旦調(diào)用堆棧中沒有任何東西,它就會去獲取事件隊列中的東西。

我們不必等待像setTimeout,fetch或其他東西這樣的函數(shù)來完成自己的工作,因為它們是由瀏覽器提供的,并且它們運行在自己的線程中。

ES6作業(yè)隊列

ECMAScript 2015引入了Promises使用的作業(yè)隊列概念(也在ES6 / ES2015中引入)。這是一種盡快執(zhí)行異步函數(shù)結(jié)果的方法,而不是放在調(diào)用堆棧的末尾。

在當前函數(shù)結(jié)束之前解析的Prom將在當前函數(shù)之后立即執(zhí)行。

我覺得在游樂園里過山車的比喻很好:消息隊列將你放在隊列的后面,在所有其他人的后面,你將不得不等待輪到你,而作業(yè)隊列是快速通票這可以讓你在完成上一個之后再騎一次。

JAVASCRIPT與單個線程的工作

這是Promises(和Async / await,它建立在promises上)和普通的舊異步函數(shù)setTimeout()或其他平臺API 之間的巨大差異。

javascrit的事件循環(huán)是這門語言中非常重要且基礎(chǔ)的概念。清楚的了解了事件循環(huán)的執(zhí)行順序和每一個階段的特點,可以使我們對一段異步代碼的執(zhí)行順序有一個清晰的認識,從而減少代碼運行的不確定性。合理的使用各種延遲事件的方法,有助于代碼更好的按照其優(yōu)先級去執(zhí)行。

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學習之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • Web
    Web
    +關(guān)注

    關(guān)注

    2

    文章

    1263

    瀏覽量

    69508
  • javascript
    +關(guān)注

    關(guān)注

    0

    文章

    519

    瀏覽量

    53875
收藏 人收藏

    評論

    相關(guān)推薦

    javascript:void(0) 是否影響SEO優(yōu)化

    使用 javascript:void(0) 確實可能對SEO優(yōu)化產(chǎn)生負面影響 。以下是關(guān)于 javascript:void(0) 對SEO影響的具體分析: 搜索引擎爬蟲的理解問題 搜索引擎爬蟲(如
    的頭像 發(fā)表于 12-31 16:08 ?113次閱讀

    javascript:void(0) 的作用是什么

    javascript:void(0) 在 HTML 和 JavaScript 中是一個常見的表達式,主要用來創(chuàng)建一個無操作的鏈接(通常是 標簽)或者阻止默認事件處理。具體來說,它的作用有以下幾點
    的頭像 發(fā)表于 12-31 15:55 ?111次閱讀

    socket 多線程編程實現(xiàn)方法

    在現(xiàn)代網(wǎng)絡(luò)編程中,多線程技術(shù)被廣泛應(yīng)用于提高服務(wù)器的并發(fā)處理能力。Socket編程是網(wǎng)絡(luò)通信的基礎(chǔ),而將多線程技術(shù)應(yīng)用于Socket編程,可以顯著提升服務(wù)器的性能。 多線程編程的基本概念 多
    的頭像 發(fā)表于 11-12 14:16 ?370次閱讀

    CPU線程和程序線程的區(qū)別

    CPU的線程與程序的線程在概念、作用、實現(xiàn)方式以及性能影響等方面存在顯著差異。以下是對兩者區(qū)別的詳細闡述,旨在深入探討這一技術(shù)話題。
    的頭像 發(fā)表于 09-02 11:18 ?1034次閱讀

    鴻蒙語言基礎(chǔ)類庫:ohos.convertxml xml轉(zhuǎn)換JavaScript

    轉(zhuǎn)換xml文本為JavaScript對象。
    的頭像 發(fā)表于 07-08 15:54 ?434次閱讀
    鴻蒙語言基礎(chǔ)類庫:ohos.convertxml  xml轉(zhuǎn)換<b class='flag-5'>JavaScript</b>

    探索虛擬線程:原理與實現(xiàn)

    虛擬線程的引入與優(yōu)勢 在Loom項目之前,Java虛擬機(JVM)中的線程是通過java.lang.Thread類型來實現(xiàn)的,這些線程被稱為平臺線程。 然而,平臺
    的頭像 發(fā)表于 06-24 11:35 ?312次閱讀
    探索虛擬<b class='flag-5'>線程</b>:原理與實現(xiàn)

    鴻蒙開發(fā):【線程模型】

    管理其他線程的ArkTS引擎實例,例如使用TaskPool(任務(wù)池)創(chuàng)建任務(wù)或取消任務(wù)、啟動和終止Worker線程
    的頭像 發(fā)表于 06-13 16:38 ?413次閱讀
    鴻蒙開發(fā):【<b class='flag-5'>線程</b>模型】

    鴻蒙APP開發(fā):【ArkTS類庫多線程】TaskPool和Worker的對比(2)

    創(chuàng)建Worker的線程稱為宿主線程(不一定是主線程,工作線程也支持創(chuàng)建Worker子線程),Wo
    的頭像 發(fā)表于 03-27 15:44 ?546次閱讀
    鴻蒙APP開發(fā):【ArkTS類庫多<b class='flag-5'>線程</b>】TaskPool和Worker的對比(2)

    java實現(xiàn)多線程的幾種方式

    Java實現(xiàn)多線程的幾種方式 多線程是指程序中包含了兩個或以上的線程,每個線程都可以并行執(zhí)行不同的任務(wù)或操作。Java中的多線程可以提高程序
    的頭像 發(fā)表于 03-14 16:55 ?714次閱讀

    python中5種線程鎖盤點

    線程安全是多線程或多進程編程中的一個概念,在擁有共享數(shù)據(jù)的多條線程并行執(zhí)行的程序中,線程安全的代碼會通過同步機制保證各個線程都可以正常且正確
    發(fā)表于 03-07 11:08 ?1599次閱讀
    python中5種<b class='flag-5'>線程</b>鎖盤點

    基于RTOS的應(yīng)用進程中的典型線程

    RTOS中的關(guān)鍵因素是最小的中斷延遲和最小的線程切換延遲。RTOS的價值在于它的響應(yīng)速度或可預測性,而不是它在給定時間段內(nèi)可以執(zhí)行的工作量。
    發(fā)表于 03-05 09:32 ?621次閱讀
    基于RTOS的應(yīng)用進程中的典型<b class='flag-5'>線程</b>

    什么是動態(tài)線程池?動態(tài)線程池的簡單實現(xiàn)思路

    因此,動態(tài)可監(jiān)控線程池一種針對以上痛點開發(fā)的線程池管理工具。主要可實現(xiàn)功能有:提供對 Spring 應(yīng)用內(nèi)線程池實例的全局管控、應(yīng)用運行時動態(tài)變更線程池參數(shù)以及
    的頭像 發(fā)表于 02-28 10:42 ?649次閱讀

    線程是什么的基本單位 進程與線程的本質(zhì)區(qū)別

    線程是操作系統(tǒng)中處理器調(diào)度的基本單位,它代表著獨立的執(zhí)行流。在一個進程中,可以包含多個線程,這些線程共享相同的進程資源,如內(nèi)存空間、文件描述符等。 進程是操作系統(tǒng)中運行的程序的實例,它包含了程序
    的頭像 發(fā)表于 02-02 16:30 ?947次閱讀

    什么是守護線程?守護線程的底層原理和使用示例

    大家好,今天這篇文章來梳理一下有關(guān)守護線程的相關(guān)問題,這也是之前曾經(jīng)有被問到過的面試題,在此之前我們先看一看守護線程的使用示例。
    的頭像 發(fā)表于 01-05 11:01 ?1429次閱讀
    什么是守護<b class='flag-5'>線程</b>?守護<b class='flag-5'>線程</b>的底層原理和使用示例

    mcu線程和進程的區(qū)別是什么

    MCU線程和進程是嵌入式系統(tǒng)中常見的并行執(zhí)行的概念,它們之間有許多區(qū)別,包括線程與進程的定義、資源管理、通信機制、執(zhí)行方式等等。下面將詳細介紹MCU線程和進程的區(qū)別。 一、定義與概念 MCU
    的頭像 發(fā)表于 01-04 10:45 ?763次閱讀