Heim >Web-Frontend >js-Tutorial >Beginnen Sie mit einem Web Worker

Beginnen Sie mit einem Web Worker

WBOY
WBOYOriginal
2023-09-04 12:49:011207Durchsuche

Eines der vielen Designziele der JavaScript-Sprache besteht darin, Single-Threaded und einfach zu bleiben. Obwohl ich zugeben muss, dass es angesichts der Natur der Sprachstrukturen alles andere als einfach ist! Was wir jedoch mit „Single-Threaded“ meinen, ist, dass es in JavaScript nur einen Steuerungsthread gibt. Leider kann Ihre JavaScript-Engine jeweils nur eine Aufgabe ausführen.

Klingt das nicht zu restriktiv für die Nutzung des Multi-Core-Prozessors im Leerlauf Ihres Computers? HTML5 verspricht, das alles zu ändern.


Single-Threaded-Modell für JavaScript

Web Worker leben in einer eingeschränkten Welt ohne Zugriff auf das DOM, da das DOM nicht threadsicher ist.

Eine Denkrichtung betrachtet die Single-Threaded-Natur von JavaScript als Vereinfachung, eine andere sieht darin eine Einschränkung. Die letztere Gruppe hat einen sehr guten Punkt, insbesondere da moderne Webanwendungen in großem Umfang JavaScript nutzen, um UI-Ereignisse zu verarbeiten, serverseitige APIs abzufragen oder abzufragen, große Datenmengen zu verarbeiten und das DOM basierend auf Serverantworten zu manipulieren. p>

In der Lage zu sein, so viele Vorgänge in einem einzigen Steuerungsthread auszuführen und gleichzeitig eine reaktionsfähige Benutzeroberfläche aufrechtzuerhalten, ist oft eine schwierige Aufgabe und zwingt Entwickler dazu, auf Hacks und Problemumgehungen (z. B. die Verwendung von setTimeout(), setInterval(),或者使用XMLHttpRequest und DOM-Ereignissen) zurückzugreifen, um Parallelität zu erreichen. Es ist jedoch erwähnenswert, dass diese Technologien sicherlich eine Möglichkeit für asynchrone Aufrufe bieten, aber nicht blockierend bedeutet nicht unbedingt gleichzeitig. John Resig erklärt in seinem Blog, warum man nichts parallel ausführen kann.

Einschränkungen

Wenn Sie JavaScript schon länger verwenden, ist Ihnen höchstwahrscheinlich das folgende lästige Dialogfeld begegnet, das besagt, dass die Ausführung bestimmter Skripts zu lange dauert. Ja, fast jedes Mal, wenn Ihre Seite nicht mehr reagiert, kann die Ursache auf einen JavaScript-Code zurückgeführt werden.

从 Web Worker 开始

Hier sind einige Gründe, warum Ihr Browser beim Ausführen von Skripten hängen bleiben kann:

  • Übermäßige DOM-Operationen: DOM-Operationen sind wahrscheinlich die teuersten Operationen in JavaScript. Daher ist Ihr Skript aufgrund der großen Anzahl an DOM-Manipulationsvorgängen ein guter Kandidat für eine Umgestaltung.
  • Schleifen ohne Ende: Es schadet nie, Ihren Code nach komplexen verschachtelten Schleifen zu durchsuchen. Diese leisten tendenziell mehr Arbeit, als tatsächlich benötigt wird. Vielleicht finden Sie eine andere Lösung, die die gleiche Funktionalität bietet.
  • Kombinieren Sie beides: Das Schlimmste, was wir tun können, ist, das DOM wiederholt in einer Schleife zu aktualisieren, wenn es elegantere Lösungen gibt (z. B. die Verwendung eines DocumentFragments).

Netzwerkarbeiter kommen zur Rettung

...nicht blockierend bedeutet nicht unbedingt gleichzeitig...

Dank HTML5 und Web Workers können Sie jetzt einen neuen Thread erzeugen – und das für echte Asynchronität. Der neue Arbeitsthread kann im Hintergrund ausgeführt werden, während der Hauptthread UI-Ereignisse verarbeitet, selbst wenn der Arbeitsthread gerade mit der Verarbeitung großer Datenmengen beschäftigt ist. Beispielsweise können Mitarbeiter große JSON-Strukturen verarbeiten, um wertvolle Informationen für die Anzeige in der Benutzeroberfläche zu extrahieren. Aber genug von meinem Geschwafel; schauen wir uns einen echten Code an.

Personal anlegen

Normalerweise befindet sich der Code für den Web Worker in einer separaten JavaScript-Datei. Der übergeordnete Thread erstellt einen neuen Worker, indem er den URI der Skriptdatei im Worker-Konstruktor angibt, der die JavaScript-Datei asynchron lädt und ausführt.

var primeWorker = new Worker('prime.js');

Starten Sie einen Worker-Thread

Um einen Arbeitsthread zu starten, sendet der übergeordnete Thread eine Nachricht wie folgt an den Arbeitsthread:

var current = $('#prime').attr('value');
primeWorker.postMessage(current);

Übergeordnete Seite kann postMessage API 与工作人员通信,该 API 也用于跨源消息传递。除了向工作程序发送原始数据类型之外,postMessage API verwenden, unterstützt auch die Übergabe der JSON-Struktur. Sie können jedoch keine Funktionen übergeben, da diese möglicherweise Verweise auf das zugrunde liegende DOM enthalten.

Der übergeordnete Thread und der Arbeitsthread verfügen über einen eigenen unabhängigen Bereich. Hin- und hergeleitete Nachrichten werden kopiert und nicht geteilt.

Unter der Haube werden diese Nachrichten am Arbeitsthread serialisiert und dann am Empfangsende deserialisiert. Daher wird davon abgeraten, große Datenmengen an Arbeitsthreads zu senden.

Der übergeordnete Thread kann auch einen Rückruf registrieren, um auf Nachrichten zu warten, die vom Arbeitsthread nach der Ausführung seiner Aufgaben zurückgesendet werden. Dadurch kann der übergeordnete Thread die erforderlichen Aktionen ausführen (z. B. das Aktualisieren des DOM), nachdem der Arbeitsthread seine Aufgabe erledigt hat. Schauen Sie sich diesen Code an:

primeWorker.addEventListener('message', function(event){
    console.log('Receiving from Worker: '+event.data);
    $('#prime').html( event.data );
});

event Das Objekt enthält zwei wichtige Eigenschaften:

  • <code><b>target</b>target
  • : Wird verwendet, um den Mitarbeiter zu identifizieren, der die Nachricht sendet. Dies ist vor allem in Umgebungen mit mehreren Mitarbeitern nützlich.
  • <b>data</b>
  • data
: Die Nachricht, die der Worker an seinen übergeordneten Thread zurücksendet.

prime.js 中,并注册从其父级接收的 message 事件。它还使用相同的 postMessage Der

Worker selbst ist in der 🎜 API enthalten, um mit dem übergeordneten Thread zu kommunizieren. 🎜
self.addEventListener('message',  function(event){
    var currPrime = event.data, nextPrime;
    setInterval( function(){

    nextPrime = getNextPrime(currPrime);
    postMessage(nextPrime);	
    currPrime = nextPrime;

    }, 500);
});

网络工作者生活在一个受限且线程安全的环境中。

在此示例中,我们只需找到下一个最大素数,然后重复将结果发送回父线程,父线程又用新值更新 UI。在工作者上下文中, selfthis 均指全局范围。 Worker 可以为 message 事件添加事件侦听器,也可以定义 onmessage 处理程序来侦听父线程发送的任何消息。

查找下一个素数的任务显然不是工作人员的理想用例,但在这里选择它是为了演示传递消息的概念。随后,我们确实探索了使用 Web Worker 真正能带来好处的可能且实际的用例。

终止员工

工人是资源密集型的;它们是操作系统级线程。因此,您不想创建大量工作线程,并且应该在 Web Worker 完成工作后终止它。工人可以终止自己,如下所示:

self.close();

或者父线程可以终止工作线程:

primeWorker.terminate();

安全和限制

在工作脚本中,我们无法访问许多重要的 JavaScript 对象,例如 documentwindowconsoleparent,最重要的是无法访问 DOM。没有 DOM 访问权限并且无法更新页面确实听起来限制太多,但这是一个重要的安全设计决策。想象一下,如果多个线程尝试更新同一元素,可能会造成严重破坏。因此,网络工作者生活在一个受限且线程安全的环境中。

话虽如此,您仍然可以使用worker来处理数据并将结果返回到主线程,然后主线程可以更新DOM。尽管他们被拒绝访问一些非常重要的 JavaScript 对象,但工作人员可以使用一些函数,例如 setTimeout()/clearTimeout()setInterval()/clearInterval()navigator 等。还可以在工作器内使用 XMLHttpRequestlocalStorage 对象。

同源限制

在工作者上下文中, selfthis 均指全局范围。

为了与服务器通信,工作人员必须遵循同源策略。例如,托管在 http://www.example.com/ 上的脚本无法访问 https://www.example.com/ 上的脚本。即使主机名相同,同源策略也规定协议也必须相同。通常,这不是问题。您很可能正在编写工作程序、客户端,并从同一域为它们提供服务,但了解限制总是有用的。

Google Chrome 的本地访问问题

Google Chrome 对本地访问工作程序设置了限制,因此您将无法在本地设置上运行这些示例。如果您想使用 Chrome,则必须在某个服务器上托管这些文件,或者在从命令行启动 Chrome 时使用 --allow-file-access-from-files 标志。对于 OS X,按如下方式启动 chrome:

$ /Applications/Google\ Chrome.app/Contents/MacOS/Google\ Chrome --allow-file-access-from-files

但是,不建议在生产环境中使用此标志。因此,最好的选择是将这些文件托管在网络服务器上,并在任何支持的浏览器中测试您的网络工作人员。

调试 Worker 和错误处理

无法访问 console 使得这有点不简单,但是借助 Chrome 开发工具,人们可以像调试任何其他 JavaScript 代码一样调试工作代码。

从 Web Worker 开始

要处理 Web Worker 抛出的任何错误,您可以侦听 error 事件,该事件会填充 ErrorEvent 对象。您可以检查该对象以了解错误的详细原因。

primeWorker.addEventListener('error', function(error){
    console.log(' Error Caused by worker: '+error.filename
        + ' at line number: '+error.lineno
        + ' Detailed Message: '+error.message);
});

多个工作线程

虽然多个工作线程在它们之间划分工作是很常见的,但需要注意的是。官方规范指定这些工作人员相对重量级,预计将是在后台运行的长期脚本。 Web Worker 不适合大量使用,因为它们的启动性能成本和每个实例的内存成本都很高。

共享工作人员简介

该规范概述了两种类型的工作人员:专用工作人员和共享工作人员。到目前为止,我们已经看到了敬业工人的例子。它们直接链接到其创建者脚本/页面,因为它们与创建它们的脚本/页面具有一对一的关系。另一方面,共享工作人员可以在同一个来源的所有页面之间共享(即:同一来源的所有页面或脚本都可以与共享工作人员通信)。

要创建共享工作线程,只需将脚本的 URL 或工作线程的名称传递给 SharedWorker 构造函数即可。

共享工作程序使用方式的主要区别在于,它们与 port 相关联,以跟踪访问它们的父脚本。

以下代码片段创建一个共享工作线程,注册回调以监听该工作线程发布的任何消息,并将消息发布到共享工作线程:

var sharedWorker = new SharedWorker('findPrime.js');
sharedWorker.port.onmessage = function(event){
    ...
}

sharedWorker.port.postMessage('data you want to send');

类似地,工作人员可以侦听 connect 事件,当新客户端尝试连接到工作人员时会收到该事件,然后相应地向其发送消息。

onconnect = function(event) {
    // event.source contains the reference to the client's port
    var clientPort = event.source;
    // listen for any messages send my this client
    clientPort.onmessage = function(event) {
        // event.data contains the message send by client
        var data = event.data;
        ....
        // Post Data after processing
        clientPort.postMessage('processed data');
    }
};

由于它们的共享性质,您可以在同一应用程序的不同选项卡中维护相同的状态,因为不同选项卡中的两个页面使用相同的共享工作脚本来维护和报告状态。有关共享工作人员的更多详细信息,我鼓励您阅读规范。


实际用例

Web Worker 不适合大量使用,因为它们的启动性能成本很高,每个实例的内存成本也很高。

现实生活中的场景可能是,您被迫处理同步第三方 API,该 API 强制主线程在继续执行下一条语句之前等待结果。在这种情况下,您可以将此任务委托给新生成的工作线程,以利用异步功能为您带来好处。

Web 工作人员还擅长轮询情况,在这种情况下,您可以在后台连续轮询目标,并在一些新数据到达时将消息发布到主线程。

您可能还需要处理服务器返回的大量数据。传统上,处理大量数据会对应用程序的响应能力产生负面影响,从而使用户体验变得不可接受。更优雅的解决方案是将处理工作分配给多个工作人员来处理数据的非重叠部分。

其他用例可能是在多个网络工作人员的帮助下分析视频或音频源,每个工作人员都处理问题的预定义部分。


结论

想象一下在单线程环境中与多个线程相关的强大功能。

与 HTML5 规范中的许多内容一样,Web Worker 规范也在不断发展。如果您打算成为网络工作者,那么看看规范不会有什么坏处。

对于使用当前版本的 Chrome、Safari 和 Firefox 的专业工作人员来说,跨浏览器支持相当不错。即使是 IE 也没有落后太多,IE10 占据了主导地位。但是,仅当前版本的 Chrome 和 Safari 支持共享工作线程。令人惊讶的是,Android 4.0 中提供的最新版本的 Android 浏览器不支持 Web Worker,尽管在 2.1 版本中支持了 Web Worker。 Apple 还从 iOS 5.0 开始提供了 Web Worker 支持。

想象一下在单线程环境中与多线程相关的强大功能。可能性是无限的!

Das obige ist der detaillierte Inhalt vonBeginnen Sie mit einem Web Worker. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Stellungnahme:
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn