摘要:最后,提供個(gè)正確使用的場(chǎng)景。異步編程的一個(gè)很好的用例就請(qǐng)求。這意味著異步函數(shù)只能解決一小部分語(yǔ)言單線程中的局限性問(wèn)題。中有類(lèi)似的集群子進(jìn)程概念,他們也是多線程但是和還是有區(qū)別。可用的特性由于的多線程特性,工作者只能訪問(wèn)特性的一個(gè)子集。
這是專(zhuān)門(mén)探索 JavaScript 及其所構(gòu)建的組件的系列文章的第7篇。
想閱讀更多優(yōu)質(zhì)文章請(qǐng)猛戳GitHub博客,一年百來(lái)篇優(yōu)質(zhì)文章等著你!
如果你錯(cuò)過(guò)了前面的章節(jié),可以在這里找到它們:
JavaScript是如何工作的:引擎,運(yùn)行時(shí)和調(diào)用堆棧的概述!
JavaScript是如何工作的:深入V8引擎&編寫(xiě)優(yōu)化代碼的5個(gè)技巧!
JavaScript如何工作:內(nèi)存管理+如何處理4個(gè)常見(jiàn)的內(nèi)存泄漏 !
JavaScript是如何工作的:事件循環(huán)和異步編程的崛起+ 5種使用 async/await 更好地編碼方式!
JavaScript是如何工作: 深入探索 websocket 和HTTP/2與SSE +如何選擇正確的路徑!
JavaScript是如何工作的:與 WebAssembly比較 及其使用場(chǎng)景 !
這次我們會(huì)逐步講解 Web Workers,先說(shuō)個(gè)簡(jiǎn)單的概念,接著討論不同類(lèi)型的 Web Workers,他們的組成部分是如何一起工作的,以及不同場(chǎng)景下它們各自優(yōu)勢(shì)和限制。最后,提供5個(gè)正確使用 Web Workers 的場(chǎng)景。
正如我們前面文章討論的那樣,你應(yīng)該知道 JavaScript 語(yǔ)言采用的是單線程模型。然而,JavaScript 也為開(kāi)發(fā)人員提供了編寫(xiě)異步代碼的機(jī)會(huì)。
異步編程的局限性以前的文章討論過(guò)異步編程,以及應(yīng)該在什么時(shí)候使用它。
異步編程可以讓UI界面是響應(yīng)式(渲染速度快)的,通過(guò)"代碼調(diào)度",讓需要請(qǐng)求時(shí)間的代碼先放到在 event loop中晚一點(diǎn)再執(zhí)行,這樣就允許UI先行渲染展示。
異步編程的一個(gè)很好的用例就 AJAX 請(qǐng)求。由于請(qǐng)求可能花費(fèi)大量時(shí)間,因此可以使用異步請(qǐng)求,在客戶端等待響應(yīng)的同時(shí)還可以執(zhí)行其他代碼。
然而,這帶來(lái)了一個(gè)問(wèn)題——請(qǐng)求是由瀏覽器的WEB API處理的,但是如何使其他代碼是異步的呢?例如,如果成功回調(diào)中的代碼非常占用CPU:
var result = performCPUIntensiveCalculation();
如果 performCPUIntensiveCalculation 不是一個(gè)HTTP請(qǐng)求而是一個(gè)阻塞代碼(比如一個(gè)內(nèi)容很多的for loop循環(huán)),就沒(méi)有辦法及時(shí)清空事件循環(huán),瀏覽器的 UI 渲染就會(huì)被阻塞,頁(yè)面無(wú)法及時(shí)響應(yīng)給用戶。
這意味著異步函數(shù)只能解決一小部分 JavaScript 語(yǔ)言單線程中的局限性問(wèn)題。
在某些情況下,可以使用 setTimeout 對(duì)長(zhǎng)時(shí)間運(yùn)行的計(jì)算阻塞的,可以使用 setTimeout暫時(shí)放入異步隊(duì)列中,從讓頁(yè)面得到更快的渲染。例如,通過(guò)在多帶帶的 setTimeout 調(diào)用中批處理復(fù)雜的計(jì)算,可以將它們放在事件循環(huán)中多帶帶的“位置”上,這樣可以爭(zhēng)取為 UI 渲染/響應(yīng)的執(zhí)行時(shí)間。
看一個(gè)簡(jiǎn)單的函數(shù),計(jì)算一個(gè)數(shù)字?jǐn)?shù)組的平均值:
以下是重寫(xiě)上述代碼并“模擬”異步性的方法:
function averageAsync(numbers, callback) { var len = numbers.length, sum = 0; if (len === 0) { return 0; } function calculateSumAsync(i) { if (i < len) { // Put the next function call on the event loop. setTimeout(function() { sum += numbers[i]; calculateSumAsync(i + 1); }, 0); } else { // The end of the array is reached so we"re invoking the callback. callback(sum / len); } } calculateSumAsync(0); }
使用setTimeout函數(shù),該函數(shù)將在事件循環(huán)中進(jìn)一步添加計(jì)算的每個(gè)步驟。在每次計(jì)算之間,將有足夠的時(shí)間進(jìn)行其他計(jì)算,從而可以讓瀏覽器進(jìn)行渲染。
Web Worker 可以解決這個(gè)問(wèn)題HTML5為我們帶來(lái)了很多新的東西,包括:
SSE(我們?cè)谇耙黄恼轮幸呀?jīng)描述并與WebSockets進(jìn)行了比較)
Geolocation
Application cache
Local Storage
Drag and Drop
Web Workers
Web Worker 概述Web Worker 的作用,就是為 JavaScript 創(chuàng)造多線程環(huán)境,允許主線程創(chuàng)建 Worker 線程,將一些任務(wù)分配給后者運(yùn)行。在主線程運(yùn)行的同時(shí),Worker 線程在后臺(tái)運(yùn)行,兩者互不干擾。等到 Worker 線程完成計(jì)算任務(wù),再把結(jié)果返回給主線程。這樣的好處是,一些計(jì)算密集型或高延遲的任務(wù),被 Worker 線程負(fù)擔(dān)了,主線程(通常負(fù)責(zé) UI 交互)就會(huì)很流暢,不會(huì)被阻塞或拖慢。
你可能會(huì)問(wèn):“JavaScript不是一個(gè)單線程的語(yǔ)言嗎?”
事實(shí)上 JavaScript 是一種不定義線程模型的語(yǔ)言。Web Workers 不是 JavaScript 的一部分,而是可以通過(guò) JavaScript 訪問(wèn)的瀏覽器特性。歷史上,大多數(shù)瀏覽器都是單線程的(當(dāng)然,這已經(jīng)改變了),大多數(shù) JavaScript 實(shí)現(xiàn)都入發(fā)生在瀏覽器中。Web Workers 不是在 Node.JS 中實(shí)現(xiàn)的。Node.js 中有類(lèi)似的集群(cluster)、子進(jìn)程概念(child_process),他們也是多線程但是和 Web Workers 還是有區(qū)別 。
值得注意的是,規(guī)范 中提到了三種類(lèi)型的 Web Workers:
專(zhuān)用 Workers (Dedicated Workers)
共享 Workers (Shared Workers)
服務(wù) Workers (Service workers)
Dedicated Workers專(zhuān)用 Workers 只能被創(chuàng)建它的頁(yè)面訪問(wèn),并且只能與它通信。以下是瀏覽器支持的情況:
Shared Workers共享 Workers 在同一源(origin)下面的各種進(jìn)程都可以訪問(wèn)它,包括:iframes、瀏覽器中的不同tab頁(yè)(一個(gè)tab頁(yè)就是一個(gè)多帶帶的進(jìn)程,所以Shared Workers可以用來(lái)實(shí)現(xiàn) tab 頁(yè)之間的交流)、以及其他的共享 Workers。以下是瀏覽器支持的情況:
Service workersService Worker 功能:
后臺(tái)消息傳遞
網(wǎng)絡(luò)代理,轉(zhuǎn)發(fā)請(qǐng)求,偽造響應(yīng)
離線緩存
消息推送
在目前階段,Service Worker 的主要能力集中在網(wǎng)絡(luò)代理和離線緩存上。具體的實(shí)現(xiàn)上,可以理解為 Service Worker 是一個(gè)能在網(wǎng)頁(yè)關(guān)閉時(shí)仍然運(yùn)行的 Web Worker。以下是瀏覽器支持的情況:
本文主要討論 專(zhuān)用 Workers,沒(méi)有特別聲明的話,Web Workers、Workers都是指代的專(zhuān)用 Workers。
Web Workers 是如何工作Web Workers 一般通過(guò)腳本為 .js 文件來(lái)構(gòu)建,在頁(yè)面中還通過(guò)了一些異步的 HTTP 請(qǐng)求,這些請(qǐng)求是完全被隱藏了的,你只需要調(diào)用 Web Worker API.
Worker 利用類(lèi)線程間消息傳遞來(lái)實(shí)現(xiàn)并行性。它們保證界面的實(shí)時(shí)性、高性能和響應(yīng)性呈現(xiàn)給用戶。
Web Workers 在瀏覽器中的一個(gè)獨(dú)立線程中運(yùn)行。因此,它們執(zhí)行的代碼需要包含在一個(gè)多帶帶的文件中。這一點(diǎn)很重要,請(qǐng)記??!
讓我們看看基本 Workers 是如何創(chuàng)建的:
var worker = new Worker("task.js");
Worker() 構(gòu)造函數(shù)的參數(shù)是一個(gè)腳本文件,該文件就是 Worker 線程所要執(zhí)行的任務(wù)。由于 Worker 不能讀取本地文件,所以這個(gè)腳本必須來(lái)自網(wǎng)絡(luò)。如果下載沒(méi)有成功(比如404錯(cuò)誤),Worker 就會(huì)默默地失敗。
為了啟動(dòng)創(chuàng)建的 Worker,需要調(diào)用 postMessage 方法:
worker.postMessage();Web Worker 通信
為了在 Web Worker 和創(chuàng)建它的頁(yè)面之間進(jìn)行通信,需要使用 postMessage 方法或 Broadcast Channel。
postMessage 方法新瀏覽器支持JSON對(duì)象作為方法的第一個(gè)參數(shù),而舊瀏覽器只支持字符串。
來(lái)看一個(gè)示例,通過(guò)將 JSON 對(duì)象作為一個(gè)更“復(fù)雜”的示例傳遞,創(chuàng)建 Worker 的頁(yè)面如何與之通信。傳遞字符串跟傳遞對(duì)象的方式也是一樣的。
讓我們來(lái)看看下面的 HTML 頁(yè)面(或者更準(zhǔn)確地說(shuō)是它的一部分):
然后這是 worker 中的 js 代碼:
self.addEventListener("message", function(e) { var data = e.data; switch (data.cmd) { case "average": var result = calculateAverage(data); // 從數(shù)值數(shù)組中計(jì)算平均值的函數(shù) self.postMessage(result); break; default: self.postMessage("Unknown command"); } }, false);
當(dāng)單擊該按鈕時(shí),將從主頁(yè)調(diào)用 postMessage。postMessage 行將 JSON 對(duì)象傳給Worker。Worker 通過(guò)定義的消息處理程序監(jiān)聽(tīng)并處理該消息。
當(dāng)消息到達(dá)時(shí),實(shí)際的計(jì)算在worker中執(zhí)行,而不會(huì)阻塞事件循環(huán)。Worker 檢查傳遞的事件參數(shù) e,像執(zhí)行 JavaScript 函數(shù)一樣,處理完成后,把結(jié)果傳回給主頁(yè)。
在 Worker 作用域中,this 和 self 都指向 Worker 的全局作用域。
有兩種方法可以停止 Worker:從主頁(yè)調(diào)用 worker.terminate() 或在 worker 內(nèi)部調(diào)用 self.close()。Broadcast Channel
Broadcast Channel API 允許同一原始域和用戶代理下的所有窗口,iFrames 等進(jìn)行交互。也就是說(shuō),如果用戶打開(kāi)了同一個(gè)網(wǎng)站的的兩個(gè)標(biāo)簽窗口,如果網(wǎng)站內(nèi)容發(fā)生了變化,那么兩個(gè)窗口會(huì)同時(shí)得到更新通知。
還是不明白?就拿 Facebook 作為例子吧,假如你現(xiàn)在已經(jīng)打開(kāi) 了Facebook 的一個(gè)窗口,但是你此時(shí)還沒(méi)有登錄,此時(shí)你又打開(kāi)另外一個(gè)窗口進(jìn)行登錄,那么你就可以通知其他窗口/標(biāo)簽頁(yè)去告訴它們一個(gè)用戶已經(jīng)登錄了并請(qǐng)求它們進(jìn)行相應(yīng)的頁(yè)面更新。
// Connection to a broadcast channel var bc = new BroadcastChannel("test_channel"); // Example of sending of a simple message bc.postMessage("This is a test message."); // Example of a simple event handler that only // logs the message to the console bc.onmessage = function (e) { console.log(e.data); } // Disconnect the channel bc.close()
可以從下面這張圖,在視覺(jué)上來(lái)清晰地感受 Broadcast Channel:
Broadcast Channel 瀏覽器支持比較有限:
消息的大小有兩種方式發(fā)送消息給Web Workers:
復(fù)制消息:消息被序列化、復(fù)制、發(fā)送,然后在另一端反序列化。頁(yè)面和 Worker 不共享相同的實(shí)例,因此最終的結(jié)果是每次傳遞都會(huì)創(chuàng)建一個(gè)副本大多數(shù)瀏覽器,在兩邊都是使用的JSON對(duì)值進(jìn)行編碼和解碼,這樣對(duì)數(shù)據(jù)的解碼、編碼操作,勢(shì)必會(huì)增加消息傳輸過(guò)程的時(shí)間開(kāi)銷(xiāo)。信息越大,發(fā)送的時(shí)間就越長(zhǎng)。
傳遞消息:這意味著原始發(fā)送方在一旦發(fā)送后不能再使用它。傳輸數(shù)據(jù)幾乎是瞬間的,這種傳輸方式的局限性在于只能用 ArrayBuffer 類(lèi)型來(lái)傳遞。
Web Workers 可用的特性由于 JavaScript的多線程特性,Web工作者只能訪問(wèn)JavaScript特性的一個(gè)子集。以下是它的一些特點(diǎn):
Web Workers 由于具有多線程特性,因此只能訪問(wèn) JavaScript 特性的子集。 以下是可使用特性列表:
navigator 對(duì)象
location 對(duì)象(只讀)
MLHttpRequest
setTimeout()/clearTimeout() and setInterval()/clearInterval()
應(yīng)用緩存(Application Cache)
使用 importScripts() 導(dǎo)入外部腳本
創(chuàng)建其他的 Web Workers
Web Workers 的局限性遺憾的是,Web Workers 無(wú)法訪問(wèn)一些非常關(guān)鍵的 JavaScript 特性:
DOM(它會(huì)造成線程不安全)
window 對(duì)象
document 對(duì)象
parent 對(duì)象
這意味著 Web Worker 不能操作 DOM (因此也不能操作 UI)。有時(shí)這可能很棘手,但是一旦你了解了如何正確使用 Web Workers,你就會(huì)開(kāi)始將它們作為多帶帶的“計(jì)算機(jī)”使用,而所有 UI 更改都將發(fā)生在你的頁(yè)面代碼中。 Workers 將為你完成所有繁重的工作,然后一旦完成再把結(jié)果返回給 page 頁(yè)面。
處理錯(cuò)誤和 JavaScript 代碼一樣,Web workers 里拋出的錯(cuò)誤,你也需要進(jìn)行處理。當(dāng) Worker 執(zhí)行過(guò)程中如果遇到錯(cuò)誤,會(huì)觸發(fā)一個(gè) ErrorEvent 事件。接口包含了三個(gè)有用的屬性來(lái)幫忙排查問(wèn)題:
filename - 導(dǎo)致 Worker 的腳本名稱(chēng)
lineno - 發(fā)生錯(cuò)誤的行號(hào)
message - 對(duì)錯(cuò)誤的描述
例子如下:
在這里,可以看到我們創(chuàng)建了一個(gè) worker 并開(kāi)始偵聽(tīng)錯(cuò)誤事件。
在 worker 內(nèi)部(在 workerWithError.js 中),我們通過(guò)將未定義 x 乘以 2 來(lái)創(chuàng)建一個(gè)異常。異常被傳播到初始腳本,然后通過(guò)頁(yè)面監(jiān)聽(tīng) error事件,對(duì)錯(cuò)誤進(jìn)行捕獲。
5個(gè)好的 Web Workers 應(yīng)用實(shí)例到目前為止,我們已經(jīng)列出了 Web Workers 的優(yōu)點(diǎn)和局限性?,F(xiàn)在讓我們看看它們最強(qiáng)大的用例是什么:
Ray tracing(光線追蹤):光線追蹤是一種以像素為單位跟蹤光的路徑生成圖像的渲染技術(shù)。光線追蹤利用 CPU 密集型的數(shù)學(xué)計(jì)算來(lái)模擬光的路徑。其思想是模擬一些效果,如反射、折射、材料等。所有這些計(jì)算邏輯都可以添加到 Web Worker 中,以避免阻塞 UI線程。更好的是——可以很容易地在多個(gè) workers 之間(以及在多個(gè)cpu之間)分割圖像呈現(xiàn)。下面是一個(gè)使用 Web Workers 的光線追蹤的簡(jiǎn)單演示—https://nerget.com/rayjs-mt/r...。
Encryption(加密):由于對(duì)個(gè)人和敏感數(shù)據(jù)的監(jiān)管越來(lái)越嚴(yán)格,端到端加密越來(lái)越受歡迎。加密是一件非常耗時(shí)的事情,特別是如果有很多數(shù)據(jù)需要頻繁加密(例如,在發(fā)送到服務(wù)器之前)。這是一個(gè)使用 Web Worker 非常好的場(chǎng)景,因?yàn)樗恍枰L問(wèn) DOM 或任何花哨的東西——它是完成其工作的純算法。只要是在 Web Worker 中工作的,對(duì)于端用戶就是無(wú)縫的,不會(huì)影響到體驗(yàn)。
Prefetching data(預(yù)取數(shù)據(jù)):為了優(yōu)化你的網(wǎng)站或 web 應(yīng)用程序并改進(jìn)數(shù)據(jù)加載時(shí)間,你可以利用 Web Workers 提前加載和存儲(chǔ)一些數(shù)據(jù),以便在需要時(shí)稍后使用。Web Workers 在這種情況下非常棒,因?yàn)樗鼈儾粫?huì)影響應(yīng)用程序的UI,這與不使用Workers 時(shí)是不同的。
Progressive Web Apps(漸進(jìn)式Web應(yīng)用程序):這種漸進(jìn)式Web應(yīng)用程序要求,即使在用戶網(wǎng)絡(luò)不穩(wěn)定的條件下,也能夠迅速的加載。這意味著數(shù)據(jù)必須本地存儲(chǔ)在瀏覽器中。這也是 IndexDB 或類(lèi)似 api 發(fā)揮作用的地方。通常情況下,客戶端的存儲(chǔ)都是必要的,但使用起來(lái)需要不阻塞UI渲染線程,那么工作就需要在 Worker 中進(jìn)行了。不過(guò),以IndexDB 為例,它提供了一些異步的API,調(diào)用它們的話也不需要使用 web worker,但如果是同步的 API,就必須要在 Worker 中使用了。
Spell checking(拼寫(xiě)檢查):一個(gè)基本的拼寫(xiě)檢查程序的工作流程如下-程序讀取一個(gè)字典文件與一個(gè)正確拼寫(xiě)單詞列表。字典被解析為一個(gè)搜索樹(shù),以使實(shí)際的文本搜索更有效。當(dāng)一個(gè)單詞被提供給檢查器時(shí),程序檢查它是否存在于預(yù)先構(gòu)建的搜索樹(shù)中。如果在樹(shù)中沒(méi)有找到該單詞,可以通過(guò)替換替換字符并測(cè)試它是否是有效的單詞(如果是用戶想要寫(xiě)的單詞),為用戶提供替代拼寫(xiě)。所有的這些處理過(guò)程都可以在 Web Worker中進(jìn)行了,用戶可以不被阻塞的輸入詞匯和句子,Web Worker 在后臺(tái)校驗(yàn)詞匯是否正確以及提供備選詞匯。
原文:
https://blog.sessionstack.com...
代碼部署后可能存在的BUG沒(méi)法實(shí)時(shí)知道,事后為了解決這些BUG,花了大量的時(shí)間進(jìn)行l(wèi)og 調(diào)試,這邊順便給大家推薦一個(gè)好用的BUG監(jiān)控工具Fundebug。
你的點(diǎn)贊是我持續(xù)分享好東西的動(dòng)力,歡迎點(diǎn)贊!
歡迎加入前端大家庭,里面會(huì)經(jīng)常分享一些技術(shù)資源。文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/100588.html
摘要:異步編程通過(guò)在事件循環(huán)中調(diào)度要稍后執(zhí)行的部分代碼來(lái)使應(yīng)用程序能夠響應(yīng),從而允許首先執(zhí)行渲染。工作人員將通過(guò)定義的消息處理程序處理該消息。的應(yīng)用場(chǎng)景到目前為止,我們列出了的優(yōu)勢(shì)和局限性。 這一次我們將分拆Web Workers:我們將提供一個(gè)概述,討論不同類(lèi)型的workers,他們的組成部分如何共同發(fā)揮作用,以及他們?cè)诓煌闆r下提供的優(yōu)勢(shì)和局限性。最后,我們將提供5個(gè)用例,其中Web W...
摘要:的生命周期的生命周期與頁(yè)面完全分離。換句話說(shuō),這個(gè)將為這個(gè)域中的所有內(nèi)容接收事件。這不是必要的,但絕對(duì)是推薦的。新的將啟動(dòng)并且安裝事件將被移除。使用,可以很容易被劫持連接并偽造響應(yīng)。后臺(tái)同步允許延遲操作,直到用戶具有穩(wěn)定的連接。 這是專(zhuān)門(mén)探索 JavaScript 及其所構(gòu)建的組件的系列文章的第8篇。 想閱讀更多優(yōu)質(zhì)文章請(qǐng)猛戳GitHub博客,一年百來(lái)篇優(yōu)質(zhì)文章等著你! 如果你錯(cuò)過(guò)了前...
摘要:最后,我們將會(huì)介紹個(gè)的使用場(chǎng)景。異步編程的局限性前面我們了解到異步編程及其使用時(shí)機(jī)。請(qǐng)求是一個(gè)很好的異步編程的使用場(chǎng)景。整個(gè)是基于單線程環(huán)境的而部分可以突破這方面的限制。最佳使用場(chǎng)景迄今為止,我們列舉了的長(zhǎng)處及其限制。 Web Workers 分類(lèi)及 5 個(gè)使用場(chǎng)景 原文請(qǐng)查閱這里,略有刪減,本文采用知識(shí)共享署名 4.0 國(guó)際許可協(xié)議共享,BY Troland。 這是 JavaScri...
摘要:字節(jié)流這個(gè)簡(jiǎn)單的模型將數(shù)據(jù)存儲(chǔ)為長(zhǎng)度不透明的字節(jié)字符串變量,將任何形式的內(nèi)部組織留給應(yīng)用層。字節(jié)流數(shù)據(jù)存儲(chǔ)的代表例子包括文件系統(tǒng)和云存儲(chǔ)服務(wù)。使用同步存儲(chǔ)會(huì)阻塞主線程,并為應(yīng)用程序的創(chuàng)建凍結(jié)體驗(yàn)。 這是專(zhuān)門(mén)探索 JavaScript 及其所構(gòu)建的組件的系列文章的第 16 篇。 想閱讀更多優(yōu)質(zhì)文章請(qǐng)猛戳GitHub博客,一年百來(lái)篇優(yōu)質(zhì)文章等著你! 如果你錯(cuò)過(guò)了前面的章節(jié),可以在這里找到它...
閱讀 1991·2021-09-26 10:19
閱讀 3266·2021-09-24 10:25
閱讀 1654·2019-12-27 11:39
閱讀 1937·2019-08-30 15:43
閱讀 683·2019-08-29 16:08
閱讀 3515·2019-08-29 16:07
閱讀 915·2019-08-26 11:30
閱讀 1279·2019-08-26 10:41