async ialah es7. async dan await ialah penambahan baharu kepada ES7 dan merupakan penyelesaian untuk operasi asynchronous/wait boleh dikatakan sebagai gula sintaktik untuk modul bersama dan fungsi penjana, menyelesaikan kod tak segerak dengan semantik yang lebih jelas. Seperti namanya, async bermaksud "tak segerak".
Persekitaran pengendalian tutorial ini: sistem Windows 7, ECMAScript versi 6, komputer Dell G3.
Ciri tak segerak bahagian hadapan yang dicadangkan dalam ES7 (ES2017): async, tunggu.
Apa itu async/wait
async and wait ialah tambahan baharu dalam ES7 Untuk penyelesaian kepada operasi tak segerak, async/wait boleh dikatakan sebagai modul bersama dan penjanaan Gula sintaksis untuk fungsi peranti. Selesaikan kod asynchronous js dengan semantik yang lebih jelas.
Async, seperti namanya, bermaksud "asynchronous". Dan menunggu secara literal bermaksud "menunggu", yang digunakan untuk menunggu penyiapan tak segerak.
Terdapat peraturan ketat antara tak segerak dan menunggu Kedua-duanya tidak boleh dipisahkan antara satu sama lain.
Pelajar yang biasa dengan modul ko harus tahu bahawa modul ko adalah modul yang ditulis oleh Master TJ yang menggunakan fungsi penjana untuk menyelesaikan proses tak segerak Ia boleh dianggap sebagai pelaksana penjana fungsi. Async/wait ialah peningkatan kepada modul bersama Ia mempunyai pelaksana fungsi penjana terbina dalam dan tidak lagi bergantung pada modul bersama. Pada masa yang sama, async mengembalikan Janji.
Dari sudut pandangan di atas, sama ada modul co atau async/wait, Promise digunakan sebagai unit paling asas Pelajar yang tidak tahu Promise boleh terlebih dahulu mengetahui lebih lanjut tentang Promise.
Membandingkan Promise, co, async/wait
Di bawah kami menggunakan contoh mudah untuk membandingkan persamaan, perbezaan dan pertukaran antara ketiga-tiga kaedah.
Kami menggunakan pemacu nodejs mongodb untuk menanyakan pangkalan data mongodb sebagai contoh. Sebabnya ialah pemacu js mongodb telah melaksanakan Promise yang dikembalikan secara lalai, dan kami tidak perlu membungkus Promise secara berasingan.
Gunakan rantaian Promise
MongoClient.connect(url + db_name).then(db => { return db.collection('blogs'); }).then(coll => { return coll.find().toArray(); }).then(blogs => { console.log(blogs.length); }).catch(err => { console.log(err); })
Kaedah Promise kemudian() boleh mengembalikan Promise atau nilai yang disegerakkan Jika nilai yang disegerakkan dikembalikan, Nilai itu akan terbungkus dalam Janji. Dalam contoh di atas, db.collection() akan mengembalikan nilai yang disegerakkan, iaitu objek koleksi, tetapi ia akan dibungkus ke dalam Promise dan akan dihantar secara telus kepada kaedah then() seterusnya. Contoh di atas menggunakan rantai Janji. Mula-mula sambung ke pangkalan data MongoClient.connect() mengembalikan Promise, kemudian dapatkan objek pangkalan data db dalam kaedah then() dan kemudian dapatkan objek coll dan kembalikannya. Dapatkan objek coll dalam kaedah then() seterusnya, kemudian lakukan pertanyaan, kembalikan hasil pertanyaan dan panggil kaedah then() lapisan demi lapisan untuk membentuk rantai Promise. Dalam rantai Promise ini, jika pengecualian berlaku dalam mana-mana pautan, ia akan ditangkap oleh tangkapan akhir(). Boleh dikatakan kod yang ditulis menggunakan rantai Promise ini lebih elegan dan prosesnya lebih jelas daripada memanggil fungsi panggil balik lapisan demi lapisan. Mula-mula dapatkan objek pangkalan data, kemudian dapatkan objek koleksi, dan akhirnya menanyakan data. Tetapi terdapat masalah yang tidak terlalu "elegan" di sini, iaitu objek yang diperolehi oleh setiap kaedah then() ialah data yang dikembalikan oleh kaedah then() sebelumnya. Ia tidak boleh diakses merentas lapisan. Apakah maksudnya, iaitu, dalam ketiga kemudian (blog => {}), kita hanya boleh mendapatkan blog hasil pertanyaan, tetapi tidak boleh menggunakan objek db dan objek coll di atas. Pada masa ini, bagaimana jika anda ingin menutup pangkalan data db.close() selepas mencetak senarai blog? Pada masa ini, terdapat dua penyelesaian:
Pertama ialah menggunakan then() nesting. Kami mengganggu rantai Promise dan menjadikannya bersarang, sama seperti menggunakan fungsi panggil balik:
MongoClient.connect(url + db_name).then(db => { let coll = db.collection('blogs'); coll.find().toArray().then(blogs => { console.log(blogs.length); db.close(); }).catch(err => { console.log(err); }); }).catch(err => { console.log(err); })
Di sini kami menyusun dua Promise, supaya dalam operasi pertanyaan terakhir, kami boleh memanggil objek db luaran. Walau bagaimanapun, kaedah ini tidak disyorkan. Alasannya mudah, kami beralih daripada satu jenis fungsi panggilan balik neraka kepada satu jenis Neraka panggilan balik Janji.
Selain itu, kita perlu menangkap pengecualian setiap Janji, kerana Janji tidak membentuk rantai.
Terdapat cara lain, iaitu untuk lulus db dalam setiap kaedah then():
MongoClient.connect(url + db_name).then(db => { return {db:db,coll:db.collection('blogs')}; }).then(result => { return {db:result.db,blogs:result.coll.find().toArray()}; }).then(result => { return result.blogs.then(blogs => { //注意這里,result.coll.find().toArray()返回的是一個Promise,因此這里需要再解析一層 return {db:result.db,blogs:blogs} }) }).then(result => { console.log(result.blogs.length); result.db.close(); }).catch(err => { console.log(err); });
Kami lulus db dalam pengembalian setiap kaedah then() Objek dikembalikan bersama-sama dengan setiap hasil yang lain. Sila ambil perhatian bahawa tidak mengapa jika setiap hasil adalah nilai yang disegerakkan, tetapi jika ia adalah nilai Promise, setiap Promise memerlukan lapisan penghuraian tambahan.
Sebagai contoh, dalam contoh di atas, dalam objek {db:result.db,blogs:result.coll.find().toArray()}
yang dikembalikan oleh kaedah kemudian() kedua, blogs
ialah kaedah Janji () seterusnya, kami tidak boleh merujuk secara langsung nilai tatasusunan senarai blog , jadi kita perlu Mula-mula panggil kaedah then() untuk menghuraikan satu lapisan, dan kemudian kembalikan dua nilai disegerakkan db dan blog. Ambil perhatian bahawa ini melibatkan sarang Promise, tetapi Promise hanya menempatkan satu tahap then().
這種方式,也是很蛋疼的一個方式,因為如果遇到then()方法中返回的不是同步的值,而是Promise的話,我們需要多做很多工作。而且,每次都透傳一個“多余”的db對象,在邏輯上也有點(diǎn)冗余。
但除此之外,對于Promise鏈的使用,如果遇到上面的問題,好像也沒其他更好的方法解決了。我們只能根據(jù)場景去選擇一種“最優(yōu)”的方案,如果要使用Promise鏈的話。
鑒于Promise上面蛋疼的問題,TJ大神將ES6中的生成器函數(shù),用co模塊包裝了一下,以更優(yōu)雅的方式來解決上面的問題。
co搭配生成器函數(shù)
如果使用co模塊搭配生成器函數(shù),那么上面的例子可以改寫如下:
const co = require('co'); co(function* (){ let db = yield MongoClient.connect(url + db_name); let coll = db.collection('blogs'); let blogs = yield coll.find().toArray(); console.log(blogs.length); db.close(); }).catch(err => { console.log(err); });
co是一個函數(shù),將接受一個生成器函數(shù)作為參數(shù),去執(zhí)行這個生成器函數(shù)。生成器函數(shù)中使用yield
關(guān)鍵字來“同步”獲取每個異步操作的值。
上面代碼在代碼形式上,比上面使用Promise鏈要優(yōu)雅,我們消滅了回調(diào)函數(shù),代碼看起來都是同步的。除了使用co和yield有點(diǎn)怪之外。
使用co模塊,我們要將所有的操作包裝成一個生成器函數(shù),然后使用co()去調(diào)用這個生成器函數(shù)。看上去也還可以接受,但是ES的進(jìn)化是不滿足于此的,于是async/await被提到了ES7的提案。
async/await
我們先看一下使用async/await改寫上面的代碼:
(async function(){ let db = await MongoClient.connect(url + db_name); let coll = db.collection('blogs'); let blogs = await coll.find().toArray(); console.log(blogs.length); db.close(); })().catch(err => { console.log(err); });
我們對比代碼可以看出,async/await和co兩種方式代碼極為相似。co換成了async,yield換成了await。同時生成器函數(shù)變成了普通函數(shù)。這種方式在語義上更加清晰明了,async表明這個函數(shù)是異步的,同時await表示要“等待”異步操作返回值。
async函數(shù)返回一個Promise,上面的代碼其實是這樣:
let getBlogs = async function(){ let db = await MongoClient.connect(url + db_name); let coll = db.collection('blogs'); let blogs = await coll.find().toArray(); db.close(); return blogs; }; getBlogs().then(result => { console.log(result.length); }).catch(err => { console.log(err); })
我們定義getBlogs為一個async函數(shù),最后返回得到的博客列表最終會被包裝成一個Promise返回,如上,我們直接調(diào)用getBlogs().then()方法可獲取async函數(shù)返回值。
好了,上面我們簡單對比了一下三種解決異步方案,下面我們來深入了解一下async/await。
深入async/await
async返回值
async用于定義一個異步函數(shù),該函數(shù)返回一個Promise。
如果async函數(shù)返回的是一個同步的值,這個值將被包裝成一個理解resolve的Promise,等同于return Promise.resolve(value)
。
await用于一個異步操作之前,表示要“等待”這個異步操作的返回值。await也可以用于一個同步的值。
//返回一個Promise let timer = async function timer(){ return new Promise((resolve,reject) => { setTimeout(() => { resolve('500'); },500); }); } timer().then(result => { console.log(result); //500 }).catch(err => { console.log(err.message); });
//返回一個同步的值 let sayHi = async function sayHi(){ let hi = await 'hello world'; return hi; //等同于return Promise.resolve(hi); } sayHi().then(result => { console.log(result); });
上面這個例子返回是一個同步的值,字符串’hello world’,sayHi()是一個async函數(shù),返回值被包裝成一個Promise,可以調(diào)用then()方法獲取返回值。對于一個同步的值,可以使用await,也可以不使用await。效果效果是一樣的。具體用不用,看情況。
比如上面使用mongodb查詢博客那個例子,let coll = db.collection('blogs');
,這里我們就沒有用await,因為這是一個同步的值。當(dāng)然,也可以使用await,這樣會顯得代碼統(tǒng)一。雖然效果是一樣的。
async函數(shù)的異常
let sayHi = async function sayHi(){ throw new Error('出錯了'); } sayHi().then(result => { console.log(result); }).catch(err => { console.log(err.message); //出錯了 });
我們直接在async函數(shù)中拋出一個異常,由于返回的是一個Promise,因此,這個異??梢哉{(diào)用返回Promise的catch()方法捕捉到。
和Promise鏈的對比:
我們的async函數(shù)中可以包含多個異步操作,其異常和Promise鏈有相同之處,如果有一個Promise被reject()那么后面的將不會再進(jìn)行。
let count = ()=>{ return new Promise((resolve,reject) => { setTimeout(()=>{ reject('故意拋出錯誤'); },500); }); } let list = ()=>{ return new Promise((resolve,reject)=>{ setTimeout(()=>{ resolve([1,2,3]); },500); }); } let getList = async ()=>{ let c = await count(); let l = await list(); return {count:c,list:l}; } console.time('begin'); getList().then(result => { console.log(result); }).catch(err => { console.timeEnd('begin'); console.log(err); }); //begin: 507.490ms //故意拋出錯誤
如上面的代碼,定義兩個異步操作,count和list,使用setTimeout延時500毫秒,count故意直接拋出異常,從輸出結(jié)果來看,count()拋出異常后,直接由catch()捕捉到了,list()并沒有繼續(xù)執(zhí)行。
并行
使用async后,我們上面的例子都是串行的。比如上個list()和count()的例子,我們可以將這個例子用作分頁查詢數(shù)據(jù)的場景。先查詢出數(shù)據(jù)庫中總共有多少條記錄,然后再根據(jù)分頁條件查詢分頁數(shù)據(jù),最后返回分頁數(shù)據(jù)以及分頁信息。
我們上面的例子count()和list()有個“先后順序”,即我們先查的總數(shù),然后又查的列表。其實,這兩個操作并無先后關(guān)聯(lián)性,我們可以異步的同時進(jìn)行查詢,然后等到所有結(jié)果都返回時再拼裝數(shù)據(jù)即可。
let count = ()=>{ return new Promise((resolve,reject) => { setTimeout(()=>{ resolve(100); },500); }); } let list = ()=>{ return new Promise((resolve,reject)=>{ setTimeout(()=>{ resolve([1,2,3]); },500); }); } let getList = async ()=>{ let result = await Promise.all([count(),list()]); return result; } console.time('begin'); getList().then(result => { console.timeEnd('begin'); //begin: 505.557ms console.log(result); //[ 100, [ 1, 2, 3 ] ] }).catch(err => { console.timeEnd('begin'); console.log(err); });
我們將count()和list()使用Promise.all()“同時”執(zhí)行,這里count()和list()可以看作是“并行”執(zhí)行的,所耗時間將是兩個異步操作中耗時最長的耗時。
Hasil akhir ialah tatasusunan yang terdiri daripada hasil kedua-dua operasi. Kita hanya perlu mengeluarkan nilai-nilai dalam tatasusunan mengikut urutan.
[Pembelajaran yang disyorkan: tutorial video javascript]
Atas ialah kandungan terperinci Adakah async untuk es6 atau es7?. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!

Alat AI Hot

Undress AI Tool
Gambar buka pakaian secara percuma

Undresser.AI Undress
Apl berkuasa AI untuk mencipta foto bogel yang realistik

AI Clothes Remover
Alat AI dalam talian untuk mengeluarkan pakaian daripada foto.

Clothoff.io
Penyingkiran pakaian AI

Video Face Swap
Tukar muka dalam mana-mana video dengan mudah menggunakan alat tukar muka AI percuma kami!

Artikel Panas

Alat panas

Notepad++7.3.1
Editor kod yang mudah digunakan dan percuma

SublimeText3 versi Cina
Versi Cina, sangat mudah digunakan

Hantar Studio 13.0.1
Persekitaran pembangunan bersepadu PHP yang berkuasa

Dreamweaver CS6
Alat pembangunan web visual

SublimeText3 versi Mac
Perisian penyuntingan kod peringkat Tuhan (SublimeText3)

Topik panas

Dalam ES6, anda boleh menggunakan kaedah reverse() objek tatasusunan untuk mencapai pembalikan tatasusunan Kaedah ini digunakan untuk membalikkan susunan elemen dalam tatasusunan, meletakkan elemen terakhir dahulu dan elemen pertama terakhir .reverse()". Kaedah reverse() akan mengubah suai tatasusunan asal Jika anda tidak mahu mengubah suainya, anda perlu menggunakannya dengan operator pengembangan "..." dan sintaksnya ialah "[...array].reverse() ".

async ialah es7. async dan await ialah penambahan baharu kepada ES7 dan merupakan penyelesaian untuk operasi asynchronous/wait boleh dikatakan sebagai gula sintaktik untuk modul bersama dan fungsi penjana, menyelesaikan kod tak segerak dengan semantik yang lebih jelas. Seperti namanya, async bermaksud "tak segerak".

Untuk keserasian pelayar. Sebagai spesifikasi baharu untuk JS, ES6 menambah banyak sintaks dan API baharu Walau bagaimanapun, penyemak imbas moden tidak mempunyai sokongan tinggi untuk ciri baharu ES6, jadi kod ES6 perlu ditukar kepada kod ES5. Dalam alat pembangun web WeChat, babel digunakan secara lalai untuk menukar kod sintaks ES6 pembangun kepada kod ES5 yang disokong dengan baik oleh ketiga-tiga terminal, membantu pembangun menyelesaikan masalah pembangunan yang disebabkan oleh persekitaran yang berbeza hanya dalam projek Hanya konfigurasi dan semak Pilihan "ES6 hingga ES5".

Langkah-langkah: 1. Tukar dua tatasusunan untuk menetapkan jenis masing-masing, dengan sintaks "newA=new Set(a);newB=new Set(b);" 2. Gunakan has() dan filter() untuk mencari set perbezaan , dengan sintaks " new Set([...newA].filter(x =>!newB.has(x)))", elemen set perbezaan akan dimasukkan dalam koleksi set dan dikembalikan 3. Gunakan Array. daripada untuk menukar set menjadi Jenis tatasusunan, sintaks "Array.from(collection)".

Dalam es5, anda boleh menggunakan fungsi for dan indexOf() untuk mencapai deduplikasi tatasusunan Sintaks "for(i=0;i<array length;i++){a=newArr.indexOf(arr[i]);if(. a== -1){...}}". Dalam es6, anda boleh menggunakan operator spread, Array.from() dan Set untuk mengalih keluar penduaan anda perlu terlebih dahulu menukar tatasusunan menjadi objek Set untuk mengalih keluar pendua, dan kemudian menggunakan fungsi spread atau Array.from() untuk tukar objek Set kembali kepada kumpulan Just.

Dalam es6, zon mati sementara ialah ralat sintaks, yang merujuk kepada arahan let dan const yang menjadikan blok membentuk skop tertutup. Dalam blok kod, sebelum pembolehubah diisytiharkan menggunakan perintah let/const, pembolehubah tidak tersedia dan tergolong dalam "zon mati" pembolehubah sebelum pembolehubah diisytiharkan ini secara sintaksis dipanggil "zon mati sementara". ES6 menetapkan bahawa promosi pembolehubah tidak berlaku dalam zon mati sementara dan pernyataan let dan const, terutamanya untuk mengurangkan ralat masa jalan dan menghalang pembolehubah daripada digunakan sebelum ia diisytiharkan, yang mungkin membawa kepada tingkah laku yang tidak dijangka.

Dalam es6, anda boleh menggunakan atribut panjang objek tatasusunan untuk menentukan bilangan item yang terdapat dalam tatasusunan, iaitu, untuk mendapatkan bilangan elemen dalam tatasusunan ini boleh mengembalikan bilangan elemen dalam tatasusunan, hanya gunakan pernyataan "array.length" Mengembalikan nilai yang mewakili bilangan elemen objek tatasusunan, iaitu nilai panjang.

Import ES6 akan menyebabkan promosi berubah-ubah. Pengangkat boleh ubah ialah promosi pengisytiharan berubah-ubah ke awal skopnya. js perlu melalui fasa kompilasi dan pelaksanaan Semasa fasa kompilasi, semua pengisytiharan pembolehubah akan dikumpul dan pembolehubah diisytiharkan terlebih dahulu, manakala penyataan lain tidak akan mengubah susunannya, semasa fasa penyusunan, langkah pertama sudah dilaksanakan, dan bahagian kedua dilaksanakan hanya apabila pernyataan dilaksanakan dalam fasa pelaksanaan.
