Heim > Web-Frontend > js-Tutorial > Hauptteil

Detaillierte Erläuterung der neuen Funktion Async-Iteration in ES9

青灯夜游
Freigeben: 2021-04-20 09:49:06
nach vorne
2469 Leute haben es durchsucht

Detaillierte Erläuterung der neuen Funktion Async-Iteration in ES9

In ES6 wurde das Konzept der synchronen Iteration mit der Referenz des Async-Operators in ES8 eingeführt. Kann es in einer asynchronen Operation durchlaufen werden?

Heute möchte ich Ihnen etwas über die neue Funktion der asynchronen Traversierung in ES9, die asynchrone Iteration, erzählen.

Asynchrone Durchquerung


Bevor wir die asynchrone Durchquerung erklären, erinnern wir uns zunächst an die synchrone Durchquerung in ES6.

Gemäß der Definition von ES6 besteht die Iteration hauptsächlich aus drei Teilen:

1. Iterable

Schauen wir uns zunächst die Definition von Iterable an:

interface Iterable {
    [Symbol.iterator]() : Iterator;
}
Nach dem Login kopieren

Iterable bedeutet, dass in diesem Objekt übertragbare Daten vorhanden sind, und Es ist notwendig, eine Factory-Methode zu implementieren, um einen Iterator zu generieren.

2. Iterator

interface Iterator {
    next() : IteratorResult;
}
Nach dem Login kopieren

Iterator kann aus Iterable erstellt werden. Iterator ist ein Cursor-ähnliches Konzept, und auf IteratorResult kann über next zugegriffen werden.

3. IteratorResult

IteratorResult sind die Daten, die bei jedem Aufruf der nächsten Methode erhalten werden.

interface IteratorResult {
    value: any;
    done: boolean;
}
Nach dem Login kopieren

Neben einem Wert, der die abzurufenden Daten angibt, verfügt IteratorResult auch über einen Wert, der angibt, ob die Durchquerung abgeschlossen ist.

Das Folgende ist ein Beispiel für das Durchlaufen eines Arrays:

rrree

Aber das obige Beispiel durchläuft synchrone Daten, z. B. eine von der http-Seite heruntergeladene Datei, wir möchten die Datei Zeile für Zeile durchlaufen. Da das Lesen einer Datenzeile ein asynchroner Vorgang ist, beinhaltet dies einen asynchronen Datendurchlauf.

Die Methode zum Hinzufügen des asynchronen Lesens von Dateien ist readLinesFromFile, daher ist die synchrone Durchlaufmethode nicht mehr auf asynchron anwendbar:

> const iterable = ['a', 'b'];
> const iterator = iterable[Symbol.iterator]();
> iterator.next()
{ value: 'a', done: false }
> iterator.next()
{ value: 'b', done: false }
> iterator.next()
{ value: undefined, done: true }
Nach dem Login kopieren

Vielleicht denken Sie, können wir das asynchrone Lesen einer Zeile in Promise kapseln und sie dann durchlaufen? synchron?

Die Idee ist gut, aber in diesem Fall ist es unmöglich zu erkennen, ob der asynchrone Vorgang abgeschlossen ist. Die Methode ist also nicht durchführbar.

Daher führte ES9 das Konzept der asynchronen Durchquerung ein:

  • Sie können den Iterator in asynchronen Iterables über Symbol.asyncIterator abrufen.

  • Die next()-Methode des asynchronen Iterators gibt ein Promises-Objekt zurück, das IteratorResults enthält.

Schauen wir uns also die API-Definition der asynchronen Durchquerung an:

//不再适用
for (const line of readLinesFromFile(fileName)) {
    console.log(line);
}
Nach dem Login kopieren

Schauen wir uns eine Anwendung der asynchronen Durchquerung an:

interface AsyncIterable {
    [Symbol.asyncIterator]() : AsyncIterator;
}
interface AsyncIterator {
    next() : Promise<IteratorResult>;
}
interface IteratorResult {
    value: any;
    done: boolean;
}
Nach dem Login kopieren

Unter anderem konvertiert createAsyncIterable eine synchrone Iterable in eine asynchrone Iterable, was wir weiter unten tun werden Schauen wir uns in diesem Abschnitt an, wie es generiert wird.

Hier konzentrieren wir uns hauptsächlich auf die Traversaloperation von asyncIterator.

Da der Async-Operator in ES8 eingeführt wurde, können wir den obigen Code auch mit Async-Funktionen umschreiben:

const asyncIterable = createAsyncIterable([&#39;a&#39;, &#39;b&#39;]);
const asyncIterator = asyncIterable[Symbol.asyncIterator]();
asyncIterator.next()
.then(iterResult1 => {
    console.log(iterResult1); // { value: &#39;a&#39;, done: false }
    return asyncIterator.next();
})
.then(iterResult2 => {
    console.log(iterResult2); // { value: &#39;b&#39;, done: false }
    return asyncIterator.next();
})
.then(iterResult3 => {
    console.log(iterResult3); // { value: undefined, done: true }
});
Nach dem Login kopieren

Traversal of asynchronous iterable


Verwenden Sie for-of, um synchron iterable zu durchlaufen, verwenden Sie for-await -of can asynchron iterierbar durchqueren.

async function f() {
    const asyncIterable = createAsyncIterable([&#39;a&#39;, &#39;b&#39;]);
    const asyncIterator = asyncIterable[Symbol.asyncIterator]();
    console.log(await asyncIterator.next());
        // { value: &#39;a&#39;, done: false }
    console.log(await asyncIterator.next());
        // { value: &#39;b&#39;, done: false }
    console.log(await asyncIterator.next());
        // { value: undefined, done: true }
}
Nach dem Login kopieren

Beachten Sie, dass „await“ in der asynchronen Funktion platziert werden muss.

Wenn während unseres asynchronen Durchlaufs eine Ausnahme auftritt, können Sie try Catch in for-await-of verwenden, um die Ausnahme abzufangen:

async function f() {
    for await (const x of createAsyncIterable([&#39;a&#39;, &#39;b&#39;])) {
        console.log(x);
    }
}
// Output:
// a
// b
Nach dem Login kopieren

Synchronized iterable gibt synchrone Iteratoren zurück, und die nächste Methode gibt {value, done } zurück.

Wenn Sie „for-await-of“ verwenden, werden synchrone Iteratoren in asynchrone Iteratoren umgewandelt. Der zurückgegebene Wert wird dann in ein Promise umgewandelt.

Wenn der vom synchronen nächsten zurückgegebene Wert selbst ein Promise-Objekt ist, ist der asynchrone Rückgabewert immer noch dasselbe Versprechen.

Das heißt, es wird: Iterable<Promise<T>> 转换成为 AsyncIterable<T>, wie im folgenden Beispiel gezeigt:

function createRejectingIterable() {
    return {
        [Symbol.asyncIterator]() {
            return this;
        },
        next() {
            return Promise.reject(new Error(&#39;Problem!&#39;));
        },
    };
}
(async function () { 
    try {
        for await (const x of createRejectingIterable()) {
            console.log(x);
        }
    } catch (e) {
        console.error(e);
            // Error: Problem!
    }
})();
Nach dem Login kopieren

Das obige Beispiel wandelt ein synchrones Promise in ein asynchrones Promise um.

async function main() {
    const syncIterable = [
        Promise.resolve(&#39;a&#39;),
        Promise.resolve(&#39;b&#39;),
    ];
    for await (const x of syncIterable) {
        console.log(x);
    }
}
main();

// Output:
// a
// b
Nach dem Login kopieren

Das obige Beispiel konvertiert synchronisierte Konstanten in Promise. Es ist ersichtlich, dass die Ergebnisse beider gleich sind.

Generierung von asynchronem Iterable


Zurück zum obigen Beispiel: Wir verwenden createAsyncIterable(syncIterable), um syncIterable in AsyncIterable zu konvertieren.

Sehen wir uns an, wie diese Methode implementiert wird:

async function main() {
    for await (const x of [&#39;a&#39;, &#39;b&#39;]) {
        console.log(x);
    }
}
main();

// Output:
// c
// d
Nach dem Login kopieren

Im obigen Code fügen wir async vor einer gewöhnlichen Generatorfunktion ein, die einen asynchronen Generator darstellt.

Bei gewöhnlichen Generatoren wird bei jedem Aufruf der nächsten Methode ein Objekt {value,done} zurückgegeben. Dieses Objekt kapselt den Ertragswert.

Bei einem Asynchrongenerator wird jedes Mal, wenn die nächste Methode aufgerufen wird, ein Versprechensobjekt zurückgegeben, das das Objekt {value,done} enthält. Dieses Objektobjekt ist eine Kapselung des Ertragswerts.

Da ein Promise-Objekt zurückgegeben wird, müssen wir nicht warten, bis das Ergebnis der asynchronen Ausführung abgeschlossen ist, bevor wir die nächste Methode erneut aufrufen.

Wir können alle asynchronen Promise-Operationen gleichzeitig über ein Promise.all ausführen:

async function* createAsyncIterable(syncIterable) {
    for (const elem of syncIterable) {
        yield elem;
    }
}
Nach dem Login kopieren

In createAsyncIterable erstellen wir ein asynchrones Iterable aus einem synchronen Iterable.

Als nächstes schauen wir uns an, wie man aus einem asynchronen Iterable ein asynchrones Iterable erstellt.

Wir wissen aus dem vorherigen Abschnitt, dass Sie „for-await-of“ verwenden können, um asynchrone iterierbare Daten zu lesen, also können wir es wie folgt verwenden:

async function* prefixLines(asyncIterable) {
    for await (const line of asyncIterable) {
        yield &#39;> &#39; + line;
    }
}
Nach dem Login kopieren

在generator一文中,我们讲到了在generator中调用generator。也就是在一个生产器中通过使用yield*来调用另外一个生成器。

同样的,如果是在异步生成器中,我们可以做同样的事情:

async function* gen1() {
    yield &#39;a&#39;;
    yield &#39;b&#39;;
    return 2;
}
async function* gen2() {
    const result = yield* gen1(); 
        // result === 2
}

(async function () {
    for await (const x of gen2()) {
        console.log(x);
    }
})();
// Output:
// a
// b
Nach dem Login kopieren

如果在异步生成器中抛出异常,这个异常也会被封装在Promise中:

async function* asyncGenerator() {
    throw new Error(&#39;Problem!&#39;);
}
asyncGenerator().next()
.catch(err => console.log(err)); // Error: Problem!
Nach dem Login kopieren

异步方法和异步生成器


异步方法是使用async function 声明的方法,它会返回一个Promise对象。

function中的return或throw异常会作为返回的Promise中的value。

(async function () {
    return &#39;hello&#39;;
})()
.then(x => console.log(x)); // hello

(async function () {
    throw new Error(&#39;Problem!&#39;);
})()
.catch(x => console.error(x)); // Error: Problem!
Nach dem Login kopieren

异步生成器是使用 async function * 申明的方法。它会返回一个异步的iterable。

通过调用iterable的next方法,将会返回一个Promise。异步生成器中yield 的值会用来填充Promise的值。如果在生成器中抛出了异常,同样会被Promise捕获到。

async function* gen() {
    yield &#39;hello&#39;;
}
const genObj = gen();
genObj.next().then(x => console.log(x));
    // { value: &#39;hello&#39;, done: false }
Nach dem Login kopieren

本文作者:flydean程序那些事

本文链接:http://www.flydean.com/es9-async-iteration/

更多编程相关知识,请访问:编程视频!!

Das obige ist der detaillierte Inhalt vonDetaillierte Erläuterung der neuen Funktion Async-Iteration in ES9. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Verwandte Etiketten:
Quelle:flydean的博客
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
Beliebte Tutorials
Mehr>
Neueste Downloads
Mehr>
Web-Effekte
Quellcode der Website
Website-Materialien
Frontend-Vorlage