Heim Web-Frontend js-Tutorial Detaillierte Erläuterung der automatischen Ausführung des Generators in ES6

Detaillierte Erläuterung der automatischen Ausführung des Generators in ES6

Oct 19, 2018 pm 03:10 PM
es6 javascript 前端

Dieser Artikel enthält eine detaillierte Erklärung der automatischen Ausführung des Generators in ES6. Ich hoffe, dass er für Freunde hilfreich ist.

Einzelne asynchrone Aufgabe

var fetch = require('node-fetch');

function* gen(){
    var url = 'https://api.github.com/users/github';
    var result = yield fetch(url);
    console.log(result.bio);
}
Nach dem Login kopieren

Um das endgültige Ausführungsergebnis zu erhalten, müssen Sie Folgendes tun:

var g = gen();
var result = g.next();

result.value.then(function(data){
    return data.json();
}).then(function(data){
    g.next(data);
});
Nach dem Login kopieren

Führen Sie zuerst die Generatorfunktion aus Holen Sie sich das Traverser-Objekt.

Verwenden Sie dann die nächste Methode, um die erste Phase der asynchronen Aufgabe auszuführen, nämlich fetch(url).

Beachten Sie, dass, da fetch(url) ein Promise-Objekt zurückgibt, der Wert des Ergebnisses ist:

{ value: Promise { <pending> }, done: false }
Nach dem Login kopieren

Schließlich fügen wir diesem Promise-Objekt eine then-Methode hinzu und formatieren zunächst die zurückgegebenen Daten . (data.json()) und rufen Sie dann g.next auf, um die erhaltenen Daten zu übergeben, damit die zweite Phase der asynchronen Aufgabe ausgeführt werden kann und die Codeausführung abgeschlossen ist.

Mehrere asynchrone Aufgaben

Wir haben im vorherigen Abschnitt nur eine Schnittstelle aufgerufen. Wenn wir also mehrere Schnittstellen aufrufen und mehrere Ausbeuten verwenden würden, müssten wir nicht kontinuierlich verschachteln in der then-Funktion...

Werfen wir also einen Blick auf die Ausführung mehrerer asynchroner Aufgaben:

var fetch = require('node-fetch');

function* gen() {
    var r1 = yield fetch('https://api.github.com/users/github');
    var r2 = yield fetch('https://api.github.com/users/github/followers');
    var r3 = yield fetch('https://api.github.com/users/github/repos');

    console.log([r1.bio, r2[0].login, r3[0].full_name].join('\n'));
}
Nach dem Login kopieren

Um das endgültige Ausführungsergebnis zu erhalten, müssen Sie möglicherweise Folgendes schreiben:

var g = gen();
var result1 = g.next();

result1.value.then(function(data){
    return data.json();
})
.then(function(data){
    return g.next(data).value;
})
.then(function(data){
    return data.json();
})
.then(function(data){
    return g.next(data).value
})
.then(function(data){
    return data.json();
})
.then(function(data){
    g.next(data)
});
Nach dem Login kopieren

Aber ich weiß, dass du es definitiv nicht so schreiben willst...

Tatsächlich können wir es mithilfe der Rekursion so schreiben:

function run(gen) {
    var g = gen();

    function next(data) {
        var result = g.next(data);

        if (result.done) return;

        result.value.then(function(data) {
            return data.json();
        }).then(function(data) {
            next(data);
        });

    }

    next();
}

run(gen);
Nach dem Login kopieren

Der Schlüssel besteht darin, beim Yield ein Promise-Objekt zurückzugeben, diesem Promise-Objekt eine then-Methode hinzuzufügen und sie auszuführen, wenn der asynchrone Vorgang erfolgreich ist Die onFullfilled-Funktion führt dann g.next in der onFullfilled-Funktion aus und ermöglicht so dem Generator dies Setzen Sie die Ausführung fort, geben Sie dann ein Versprechen zurück, führen Sie g.next bei Erfolg aus und kehren Sie dann zurück...

Starterfunktion

Führen Sie diese Starterfunktion aus, wir formatieren die Daten in der then-Funktion data.json(), aber in allgemeineren Fällen, zum Beispiel folgt auf yield direkt ein Promise und nicht ein Promise, das von der fetch-Funktion zurückgegeben wird, da es keine JSON-Methode gibt, meldet der Code einen Fehler . Um die Vielseitigkeit zu erhöhen, haben wir es zusammen mit diesem Beispiel und Starter wie folgt geändert:

var fetch = require('node-fetch');

function* gen() {
    var r1 = yield fetch('https://api.github.com/users/github');
    var json1 = yield r1.json();
    var r2 = yield fetch('https://api.github.com/users/github/followers');
    var json2 = yield r2.json();
    var r3 = yield fetch('https://api.github.com/users/github/repos');
    var json3 = yield r3.json();

    console.log([json1.bio, json2[0].login, json3[0].full_name].join('\n'));
}

function run(gen) {
    var g = gen();

    function next(data) {
        var result = g.next(data);

        if (result.done) return;

        result.value.then(function(data) {
            next(data);
        });

    }

    next();
}

run(gen);
Nach dem Login kopieren

Solange auf yield ein Promise-Objekt folgt, können wir diese Ausführungsfunktion verwenden, um die Generatorfunktion automatisch auszuführen .

Callback-Funktion

Muss nach yield ein Promise-Objekt folgen, um die automatische Ausführung des Generators sicherzustellen? Was ist, wenn es nur eine Rückruffunktion ist? Schauen wir uns ein Beispiel an:

Zuerst simulieren wir eine normale asynchrone Anfrage:

function fetchData(url, cb) {
    setTimeout(function(){
        cb({status: 200, data: url})
    }, 1000)
}
Nach dem Login kopieren

Wir transformieren diese Funktion in:

function fetchData(url) {
    return function(cb){
        setTimeout(function(){
            cb({status: 200, data: url})
        }, 1000)
    }
}
Nach dem Login kopieren

Für eine solche Generatorfunktion:

function* gen() {
    var r1 = yield fetchData('https://api.github.com/users/github');
    var r2 = yield fetchData('https://api.github.com/users/github/followers');

    console.log([r1.data, r2.data].join('\n'));
}
Nach dem Login kopieren

Wenn Sie das Endergebnis erhalten möchten:

var g = gen();

var r1 = g.next();

r1.value(function(data) {
    var r2 = g.next(data);
    r2.value(function(data) {
        g.next(data);
    });
});
Nach dem Login kopieren

Wenn wir so schreiben, stehen wir vor dem gleichen Problem wie im ersten Abschnitt, das heißt, wenn mehrere Erträge verwendet werden, wird der Code sein eingebettet in eine Schleife. Füge es zusammen...

verwendet auch Rekursion, sodass wir es umwandeln können in:

function run(gen) {
    var g = gen();

    function next(data) {
        var result = g.next(data);

        if (result.done) return;

        result.value(next);
    }

    next();
}

run(gen);
Nach dem Login kopieren

run

Davon haben wir Sie können sehen, dass die Generatorfunktion für die automatische Ausführung einen Mechanismus erfordert, der automatisch Ausführungsrechte übergeben kann, wenn der asynchrone Vorgang Ergebnisse liefert.

Und es gibt zwei Möglichkeiten, dies zu tun.

(1) Rückruffunktion. Umschließen Sie den asynchronen Vorgang, machen Sie die Rückruffunktion verfügbar und geben Sie die Ausführungsrechte in der Rückruffunktion zurück.

(2) Versprechensobjekt. Wickeln Sie asynchrone Vorgänge in Promise-Objekte ein und verwenden Sie die then-Methode, um Ausführungsrechte zurückzugeben.

In jeder der beiden Methoden haben wir eine Run-Launcher-Funktion geschrieben. Können wir diese beiden Methoden kombinieren und eine allgemeine Run-Funktion schreiben? Versuchen wir es:

rrree

Bestimmen Sie, ob result.value ein Promise ist. Wenn nicht, führen Sie es direkt aus.

return Promise

Wir haben eine nette Starterfunktion geschrieben, die yield unterstützt, gefolgt von einer Callback-Funktion oder einem Promise-Objekt.

Nun gibt es eine Frage, über die wir nachdenken müssen: Wie erhalten wir den Rückgabewert der Generatorfunktion? Und wenn in der Generatorfunktion ein Fehler auftritt, z. B. beim Abrufen einer nicht vorhandenen Schnittstelle, wie kann dieser Fehler abgefangen werden?

Das kann man sich leicht vorstellen Promise: Wenn diese Starterfunktion ein Promise zurückgibt, können wir es zu diesem Promise-Objekt hinzufügen Funktion: Wenn alle asynchronen Vorgänge erfolgreich ausgeführt wurden, führen wir die Funktion onFullfilled aus, und wenn ein Fehler auftritt, führen wir die Funktion onRejected aus.

Wir schreiben eine Version:

// 第一版
function run(gen) {
    var gen = gen();

    function next(data) {
        var result = gen.next(data);
        if (result.done) return;

        if (isPromise(result.value)) {
            result.value.then(function(data) {
                next(data);
            });
        } else {
            result.value(next)
        }
    }

    next()
}

function isPromise(obj) {
    return 'function' == typeof obj.then;
}

module.exports = run;
Nach dem Login kopieren

Sie unterscheidet sich stark von der ersten Version:

Zuerst geben wir ein Promise zurück, wenn result.done wahr ist, wir werden auflösen( result.value) der Wert. Wenn während der Ausführung ein Fehler auftritt und abgefangen wird, lehnen wir den Grund ab.

Zweitens verwenden wir thunkToPromise, um die Rückruffunktion in ein Promise zu verpacken, und fügen dann die then-Funktion einheitlich hinzu. Es ist hier erwähnenswert, dass wir in der thunkToPromise-Funktion dem Prinzip „Error zuerst“ folgen, was bedeutet, dass wir uns mit der Situation der Rückruffunktion befassen:

// 第二版
function run(gen) {
    var gen = gen();

    return new Promise(function(resolve, reject) {

        function next(data) {
            try {
                var result = gen.next(data);
            } catch (e) {
                return reject(e);
            }

            if (result.done) {
                return resolve(result.value)
            };

            var value = toPromise(result.value);

            value.then(function(data) {
                next(data);
            }, function(e) {
                reject(e)
            });
        }

        next()
    })

}

function isPromise(obj) {
    return 'function' == typeof obj.then;
}

function toPromise(obj) {
    if (isPromise(obj)) return obj;
    if ('function' == typeof obj) return thunkToPromise(obj);
    return obj;
}

function thunkToPromise(fn) {
    return new Promise(function(resolve, reject) {
        fn(function(err, res) {
            if (err) return reject(err);
            resolve(res);
        });
    });
}

module.exports = run;
Nach dem Login kopieren

Bei Erfolg sollte der erste Parameter null zurückgeben, was darauf hinweist, dass dies der Fall ist keine Fehlerursache.

Optimierung

Basierend auf der zweiten Version haben wir den Code prägnanter und eleganter geschrieben. Der endgültige Code lautet wie folgt:

// 模拟数据请求
function fetchData(url) {
    return function(cb) {
        setTimeout(function() {
            cb(null, { status: 200, data: url })
        }, 1000)
    }
}
Nach dem Login kopieren

co

如果我们再将这个启动器函数写的完善一些,我们就相当于写了一个 co,实际上,上面的代码确实是来自于 co……

而 co 是什么? co 是大神 TJ Holowaychuk 于 2013 年 6 月发布的一个小模块,用于 Generator 函数的自动执行。

如果直接使用 co 模块,这两种不同的例子可以简写为:

// yield 后是一个 Promise
var fetch = require('node-fetch');
var co = require('co');

function* gen() {
    var r1 = yield fetch('https://api.github.com/users/github');
    var json1 = yield r1.json();
    var r2 = yield fetch('https://api.github.com/users/github/followers');
    var json2 = yield r2.json();
    var r3 = yield fetch('https://api.github.com/users/github/repos');
    var json3 = yield r3.json();

    console.log([json1.bio, json2[0].login, json3[0].full_name].join('\n'));
}

co(gen);
Nach dem Login kopieren
// yield 后是一个回调函数
var co = require('co');

function fetchData(url) {
    return function(cb) {
        setTimeout(function() {
            cb(null, { status: 200, data: url })
        }, 1000)
    }
}

function* gen() {
    var r1 = yield fetchData('https://api.github.com/users/github');
    var r2 = yield fetchData('https://api.github.com/users/github/followers');

    console.log([r1.data, r2.data].join('\n'));
}

co(gen);
Nach dem Login kopieren

是不是特别的好用?

Das obige ist der detaillierte Inhalt vonDetaillierte Erläuterung der automatischen Ausführung des Generators in ES6. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Erklärung dieser Website
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

Heiße KI -Werkzeuge

Undresser.AI Undress

Undresser.AI Undress

KI-gestützte App zum Erstellen realistischer Aktfotos

AI Clothes Remover

AI Clothes Remover

Online-KI-Tool zum Entfernen von Kleidung aus Fotos.

Undress AI Tool

Undress AI Tool

Ausziehbilder kostenlos

Clothoff.io

Clothoff.io

KI-Kleiderentferner

Video Face Swap

Video Face Swap

Tauschen Sie Gesichter in jedem Video mühelos mit unserem völlig kostenlosen KI-Gesichtstausch-Tool aus!

Heiße Werkzeuge

Notepad++7.3.1

Notepad++7.3.1

Einfach zu bedienender und kostenloser Code-Editor

SublimeText3 chinesische Version

SublimeText3 chinesische Version

Chinesische Version, sehr einfach zu bedienen

Senden Sie Studio 13.0.1

Senden Sie Studio 13.0.1

Leistungsstarke integrierte PHP-Entwicklungsumgebung

Dreamweaver CS6

Dreamweaver CS6

Visuelle Webentwicklungstools

SublimeText3 Mac-Version

SublimeText3 Mac-Version

Codebearbeitungssoftware auf Gottesniveau (SublimeText3)

WebSocket und JavaScript: Schlüsseltechnologien zur Implementierung von Echtzeitüberwachungssystemen WebSocket und JavaScript: Schlüsseltechnologien zur Implementierung von Echtzeitüberwachungssystemen Dec 17, 2023 pm 05:30 PM

WebSocket und JavaScript: Schlüsseltechnologien zur Realisierung von Echtzeit-Überwachungssystemen Einführung: Mit der rasanten Entwicklung der Internet-Technologie wurden Echtzeit-Überwachungssysteme in verschiedenen Bereichen weit verbreitet eingesetzt. Eine der Schlüsseltechnologien zur Erzielung einer Echtzeitüberwachung ist die Kombination von WebSocket und JavaScript. In diesem Artikel wird die Anwendung von WebSocket und JavaScript in Echtzeitüberwachungssystemen vorgestellt, Codebeispiele gegeben und deren Implementierungsprinzipien ausführlich erläutert. 1. WebSocket-Technologie

PHP und Vue: eine perfekte Kombination von Front-End-Entwicklungstools PHP und Vue: eine perfekte Kombination von Front-End-Entwicklungstools Mar 16, 2024 pm 12:09 PM

PHP und Vue: eine perfekte Kombination von Front-End-Entwicklungstools In der heutigen Zeit der rasanten Entwicklung des Internets ist die Front-End-Entwicklung immer wichtiger geworden. Da Benutzer immer höhere Anforderungen an das Erlebnis von Websites und Anwendungen stellen, müssen Frontend-Entwickler effizientere und flexiblere Tools verwenden, um reaktionsfähige und interaktive Schnittstellen zu erstellen. Als zwei wichtige Technologien im Bereich der Front-End-Entwicklung können PHP und Vue.js in Kombination als perfekte Waffe bezeichnet werden. In diesem Artikel geht es um die Kombination von PHP und Vue sowie um detaillierte Codebeispiele, die den Lesern helfen sollen, diese beiden besser zu verstehen und anzuwenden

JavaScript und WebSocket: Aufbau eines effizienten Echtzeit-Wettervorhersagesystems JavaScript und WebSocket: Aufbau eines effizienten Echtzeit-Wettervorhersagesystems Dec 17, 2023 pm 05:13 PM

JavaScript und WebSocket: Aufbau eines effizienten Echtzeit-Wettervorhersagesystems Einführung: Heutzutage ist die Genauigkeit von Wettervorhersagen für das tägliche Leben und die Entscheidungsfindung von großer Bedeutung. Mit der Weiterentwicklung der Technologie können wir genauere und zuverlässigere Wettervorhersagen liefern, indem wir Wetterdaten in Echtzeit erhalten. In diesem Artikel erfahren Sie, wie Sie mit JavaScript und WebSocket-Technologie ein effizientes Echtzeit-Wettervorhersagesystem aufbauen. In diesem Artikel wird der Implementierungsprozess anhand spezifischer Codebeispiele demonstriert. Wir

Einfaches JavaScript-Tutorial: So erhalten Sie den HTTP-Statuscode Einfaches JavaScript-Tutorial: So erhalten Sie den HTTP-Statuscode Jan 05, 2024 pm 06:08 PM

JavaScript-Tutorial: So erhalten Sie HTTP-Statuscode. Es sind spezifische Codebeispiele erforderlich. Vorwort: Bei der Webentwicklung ist häufig die Dateninteraktion mit dem Server erforderlich. Bei der Kommunikation mit dem Server müssen wir häufig den zurückgegebenen HTTP-Statuscode abrufen, um festzustellen, ob der Vorgang erfolgreich ist, und die entsprechende Verarbeitung basierend auf verschiedenen Statuscodes durchführen. In diesem Artikel erfahren Sie, wie Sie mit JavaScript HTTP-Statuscodes abrufen und einige praktische Codebeispiele bereitstellen. Verwenden von XMLHttpRequest

Häufig gestellte Fragen von Front-End-Interviewern Häufig gestellte Fragen von Front-End-Interviewern Mar 19, 2024 pm 02:24 PM

In Front-End-Entwicklungsinterviews decken häufige Fragen ein breites Themenspektrum ab, darunter HTML/CSS-Grundlagen, JavaScript-Grundlagen, Frameworks und Bibliotheken, Projekterfahrung, Algorithmen und Datenstrukturen, Leistungsoptimierung, domänenübergreifende Anfragen, Front-End-Engineering, Designmuster sowie neue Technologien und Trends. Interviewerfragen sollen die technischen Fähigkeiten, die Projekterfahrung und das Verständnis des Kandidaten für Branchentrends beurteilen. Daher sollten Kandidaten in diesen Bereichen umfassend vorbereitet sein, um ihre Fähigkeiten und Fachkenntnisse unter Beweis zu stellen.

Ist Django Front-End oder Back-End? Hör zu! Ist Django Front-End oder Back-End? Hör zu! Jan 19, 2024 am 08:37 AM

Django ist ein in Python geschriebenes Webanwendungs-Framework, das Wert auf schnelle Entwicklung und saubere Methoden legt. Obwohl Django ein Web-Framework ist, müssen Sie zur Beantwortung der Frage, ob Django ein Front-End oder ein Back-End ist, ein tiefes Verständnis der Konzepte von Front-End und Back-End haben. Das Front-End bezieht sich auf die Schnittstelle, mit der Benutzer direkt interagieren, und das Back-End bezieht sich auf serverseitige Programme. Sie interagieren mit Daten über das HTTP-Protokoll. Wenn das Front-End und das Back-End getrennt sind, können die Front-End- und Back-End-Programme unabhängig voneinander entwickelt werden, um Geschäftslogik bzw. interaktive Effekte sowie den Datenaustausch zu implementieren.

Erkundung der Front-End-Technologie der Go-Sprache: eine neue Vision für die Front-End-Entwicklung Erkundung der Front-End-Technologie der Go-Sprache: eine neue Vision für die Front-End-Entwicklung Mar 28, 2024 pm 01:06 PM

Als schnelle und effiziente Programmiersprache erfreut sich Go im Bereich der Backend-Entwicklung großer Beliebtheit. Allerdings assoziieren nur wenige Menschen die Go-Sprache mit der Front-End-Entwicklung. Tatsächlich kann die Verwendung der Go-Sprache für die Front-End-Entwicklung nicht nur die Effizienz verbessern, sondern Entwicklern auch neue Horizonte eröffnen. In diesem Artikel wird die Möglichkeit der Verwendung der Go-Sprache für die Front-End-Entwicklung untersucht und spezifische Codebeispiele bereitgestellt, um den Lesern ein besseres Verständnis dieses Bereichs zu erleichtern. In der traditionellen Frontend-Entwicklung werden häufig JavaScript, HTML und CSS zum Erstellen von Benutzeroberflächen verwendet

Django: Ein magisches Framework, das sowohl Front-End- als auch Back-End-Entwicklung bewältigen kann! Django: Ein magisches Framework, das sowohl Front-End- als auch Back-End-Entwicklung bewältigen kann! Jan 19, 2024 am 08:52 AM

Django: Ein magisches Framework, das sowohl Front-End- als auch Back-End-Entwicklung bewältigen kann! Django ist ein effizientes und skalierbares Webanwendungs-Framework. Es unterstützt mehrere Webentwicklungsmodelle, einschließlich MVC und MTV, und kann problemlos hochwertige Webanwendungen entwickeln. Django unterstützt nicht nur die Back-End-Entwicklung, sondern kann auch schnell Front-End-Schnittstellen erstellen und durch die Vorlagensprache eine flexible Ansichtsanzeige erreichen. Django kombiniert Front-End-Entwicklung und Back-End-Entwicklung zu einer nahtlosen Integration, sodass sich Entwickler nicht auf das Lernen spezialisieren müssen

See all articles