aboutsummaryrefslogtreecommitdiff
path: root/files/de/web/javascript/eventloop
diff options
context:
space:
mode:
authorPeter Bengtsson <mail@peterbe.com>2020-12-08 14:41:15 -0500
committerPeter Bengtsson <mail@peterbe.com>2020-12-08 14:41:15 -0500
commit4b1a9203c547c019fc5398082ae19a3f3d4c3efe (patch)
treed4a40e13ceeb9f85479605110a76e7a4d5f3b56b /files/de/web/javascript/eventloop
parent33058f2b292b3a581333bdfb21b8f671898c5060 (diff)
downloadtranslated-content-4b1a9203c547c019fc5398082ae19a3f3d4c3efe.tar.gz
translated-content-4b1a9203c547c019fc5398082ae19a3f3d4c3efe.tar.bz2
translated-content-4b1a9203c547c019fc5398082ae19a3f3d4c3efe.zip
initial commit
Diffstat (limited to 'files/de/web/javascript/eventloop')
-rw-r--r--files/de/web/javascript/eventloop/index.html138
1 files changed, 138 insertions, 0 deletions
diff --git a/files/de/web/javascript/eventloop/index.html b/files/de/web/javascript/eventloop/index.html
new file mode 100644
index 0000000000..9f8093aa57
--- /dev/null
+++ b/files/de/web/javascript/eventloop/index.html
@@ -0,0 +1,138 @@
+---
+title: Concurrency model and Event Loop
+slug: Web/JavaScript/EventLoop
+tags:
+ - Advanced
+ - Event Loop
+ - Event Management
+ - Event Queue
+ - Handling Events
+ - JavaScript
+ - events
+translation_of: Web/JavaScript/EventLoop
+---
+<div>{{JsSidebar("Advanced")}}</div>
+
+<p>JavaScript hat ein Nebenläufigkeitsmodell, auf Basis einer Eventschleife. Dieses Modell unterscheidet sich stark von Modellen aus anderen Sprachen wie C und Java.</p>
+
+<h2 id="Laufzeitkonzepte">Laufzeitkonzepte</h2>
+
+<p>Der folgenden Abschnitt erklärt ein theoretisches Modell. Moderne JavaScript-Engines implementieren und optimieren die beschriebenen Semantiken sehr stark.</p>
+
+<h3 id="Visuelle_Repräsentation">Visuelle Repräsentation</h3>
+
+<p style="text-align: center;"><img alt="Stack, heap, queue" src="/files/4617/default.svg" style="height: 270px; width: 294px;"></p>
+
+<h3 id="Stack">Stack</h3>
+
+<p>Funktionsaufrufe von einem Stack von <em>Frames.</em></p>
+
+<pre class="brush: js">function foo(b){
+ var a = 10;
+ return a + b + 11;
+}
+
+function bar(x){
+ var y = 3;
+ return foo(x * y);
+}
+
+console.log(bar(7)); // gibt 42 zurück
+</pre>
+
+<p>Beim Aufruf von <code>bar</code> wird ein erster Frame erstellt, der die Argumente und lokalen Variablen von <code>bar</code> enthält. Sobald <code>bar</code> die Funktion <code>foo</code> aufruft, wird ein zweiter Frame erstellt, der auf den ersten gelegt wird und die Argumente und lokalen Variablen von <code>foo</code> enthält. Wenn <code>foo</code> beendet wird, wird der der oberste Frame aus dem Stack entfernt (nur der <code>bar</code> Frame bleibt auf dem Stack). Wenn <code>bar</code> beendet wird, ist der Stack leer.</p>
+
+<h3 id="Heap">Heap</h3>
+
+<p>Objekte werden in einem Heap gespeichert, welcher hauptsächlich eine meist große unstrukturierte Region im Speicher ist.</p>
+
+<h3 id="Queue">Queue</h3>
+
+<p>Eine JavaScript Laufzeitumgebung benutzt eine Nachrichten-Queue, welche eine Liste von Nachrichten ist, die ausgeführt werden. Jede Nachricht hat eine Funktion, die aufgerufen wird, um die Nachricht abzuarbeiten.</p>
+
+<p>An diesem Punkt, während der {{anch("Eventschleife", "Eventschleife")}}, beginnt die Laufzeitumgebung mit der ältesten Nachricht. Dazu wird die Nachricht aus der Queue entfernt und die zugehörige Funktion mit der Nachricht als Eingabeparameter aufgerufen. Wie immer erzeugt das Aufrufen einer Funktion einen neuen Frame auf dem Stack, für den Funktionsgebrauch.</p>
+
+<p>Die Ausführung von Funktionen geht so lange weiter, bis der Stack wieder leer ist; dann wird die Eventschleife die nächste Nachricht ausführen (wenn eine vorhanden ist).</p>
+
+<h2 id="Eventschleife">Eventschleife</h2>
+
+<p>Die <strong>Eventschleife</strong> ist nach der Art und Weise, wie diese meist implementiert ist, benannt. Dies ähnelt meist der folgenden Struktur:</p>
+
+<pre class="brush: js">while (queue.waitForMessage()){
+ queue.processNextMessage();
+}</pre>
+
+<p><code>queue.waitForMessage()</code> wartet synchron auf eine eingehende Nachricht, wenn keine vorhanden ist.</p>
+
+<h3 id="Ausführungsfertigstellung">Ausführungsfertigstellung</h3>
+
+<p>Jede Nachricht wird vollständig verarbeitet, bevor eine andere Nachricht verarbeitet wird. Dies bietet einige nette Eigenschaften zum Verständnis der Ausführung ihres Programms, einschließlich der Tatsache, dass eine Funktion, wann immer sie ausgeführt wird, nicht vorzeitig verlassen werden kann. Die Funktion wird daher vollständig ausgeführt bevor irgendein anderer Code ausgeführt wird (und die Daten ändern könnte). Dies unterscheidet sich z. B. von C, wo eine Funktion in einem Thread läuft und gestoppt werden kann, um anderen Code in einem anderen Thread auszuführen.</p>
+
+<p>Dies bringt aber den Nachteil mit sich, dass während einer Nachricht, die sehr lange dauern kann, keine Nutzer-Interaktionen, wie z.B. Klicken oder Scrollen, möglich ist. Der Browser entschärft dies mit dem "Ein Skript antwortet nicht mehr" Dialog. Gute Praxis ist es daher die Dauer der Nachrichten kurz zu halten und wenn möglich in einzelne Nachrichten aufzuteilen.</p>
+
+<h3 id="Hinzufügen_von_Nachrichten">Hinzufügen von Nachrichten</h3>
+
+<p>In einem Webbrowser werden Nachrichten immer hinzugefügt, wenn ein Event auftritt und diesem ein Event Listener hinzugefügt wurde. Ist kein Listener vorhanden, geht das Event verloren. Ein Klick auf ein Element mit einem Click-Event-Handler fügt also eine Nachricht hinzu - ähnlich wie bei jedem anderen Ereignis.</p>
+
+<p>Die Funktion {{domxref("WindowTimers.setTimeout", "setTimeout")}} wird mit 2 Argumenten aufgerufen: eine Nachricht, die der Queue hinzugefügt wird, und einen Zeitwert (optional; Standardwert 0). Der Zeitwert gibt die (minimale) Zeit an, nach der die Nachricht zur Queue hinzugefügt wird. Die Nachricht wird nach dieser Zeit aufgeführt, wenn keine anderen Nachrichten ausgeführt werden. Aus diesem Grund ist der zweite Parameter die minimale und nicht die garantierte Zeit.</p>
+
+<p>Hier ist ein Beispiel, welches das Konzept demonstriert (<code>setTimeout</code> wird nicht direkt nach dem Ablaufen der Zeit aufgeführt):</p>
+
+<pre class="brush: js">const s = new Date().getSeconds();
+
+setTimeout(function() {
+ // prints out "2", meaning that the callback is not called immediately after 500 milliseconds.
+ console.log("Ran after " + (new Date().getSeconds() - s) + " seconds");
+}, 500);
+
+while(true) {
+ if(new Date().getSeconds() - s &gt;= 2) {
+ console.log("Good, looped for 2 seconds");
+ break;
+ }
+}
+</pre>
+
+<h3 id="Nullverzögerungen">Nullverzögerungen</h3>
+
+<p>Nullverzögerung bedeutet nicht, dass der Aufruf nach null Millisekunden gefeuert wird. Der Aufruf von {{domxref("WindowTimers.setTimeout", "setTimeout")}} mit einer Verzögerung von 0 Millisekunden führt die übergebene Funktion nicht nach dem gegebenen Intervall aus.</p>
+
+<p>Die Ausführung hängt von der Anzahl von wartenden Aufrufen in der Queue ab. Im Beispiel unten, wird die Nachricht ''this is just a message'' vor der Ausführung des Callback auf der Konsole ausgegeben, weil die Verzögerung die minimale Verzögerung für die Laufzeigumgebung ist, aber diese nicht garantiert werden kann.</p>
+
+<p>Im Grunde muss setTimeout warten, bis der gesamte Code für Nachrichten in der Warteschlange abgeschlossen wurde, obwohl ein bestimmtes Zeitlimit für setTimeout angegeben wurde.</p>
+
+<pre class="brush: js">(function() {
+
+ console.log('this is the start');
+
+ setTimeout(function cb() {
+ console.log('this is a msg from call back');
+ });
+
+ console.log('this is just a message');
+
+ setTimeout(function cb1() {
+ console.log('this is a msg from call back1');
+ }, 0);
+
+ console.log('this is the end');
+
+})();
+
+// "this is the start"
+// "this is just a message"
+// "this is the end"
+// note that function return, which is undefined, happens here
+// "this is a msg from call back"
+// "this is a msg from call back1"
+</pre>
+
+<h3 id="Komunikation_mehrere_Laufzeitumgebungen_zur_gleichen_Zeit">Komunikation mehrere Laufzeitumgebungen zur gleichen Zeit</h3>
+
+<p>Ein Web Worker oder ein Cross-Origin iFrame hat seinen eigenen Stack, Heap und Nachrichten Queue. Zwei unabhängige Laufzeitumgebungen können nur durch das Senden von Nachrichten über die <a href="/de/docs/Web/API/Window/postMessage"><code>postMessage</code></a> Methode kommunizieren. Diese Methode fügt der Message Queue der anderen Laufzeit eine Nachricht hinzu, wenn diese auf die Message-Events hört.</p>
+
+<h2 id="Niemals_blockierend">Niemals blockierend</h2>
+
+<p>Eine sehr interessante Eigenschaft des Eventschleifen Modells ist, dass Javascript, im Gegensatz zu vielen anderen Sprachen, niemals blockiert. Die Handhabung von I/O wird typischerweise über Events und Callback-Funktionen erledigt, so dass die Applikation, während sie auf Ergebnisse einer <a href="/de/docs/Web/API/IndexedDB_API">IndexedDB</a>- oder <a href="/de/docs/Web/API/XMLHttpRequest">XHR</a>-Anfrage wartet, weitere Benutzereingaben verarbeiten kann.</p>
+
+<p>Es gibt Ausnahmen, wie z.B. alert oder synchrone XHR, wobei es eine gute Praxis ist, diese zu vermeiden. Obacht, <a href="http://stackoverflow.com/questions/2734025/is-javascript-guaranteed-to-be-single-threaded/2734311#2734311">es existieren Ausnahmen für die Ausnahme</a> (aber diese sind für gewöhnlich eher Bugs bei der Implementierung als irgendetwas anderes).</p>