衆所周知 JavaScript 是單線程
工做,也就是隻有一個腳本執行完成後才能執行下一個腳本,兩個腳本不能同時執行,若是某個腳本耗時很長,後面的腳本都必須排隊等着,會拖延整個程序的執行。如下爲幾種異步編程方式的總結,但願與君共勉。vue
異步編程傳統的解決方案:回調函數和事件監聽ios
初始示例:假設有兩個函數, f1 和 f2,f1 是一個須要必定時間的函數。es6
function f1() { setTimeout(function(){ console.log('先執行 f1') },1000) } function f2() { console.log('再執行 f2') }
由於 f1 是一個須要必定時間的函數,因此能夠將 f2 寫成 f1 的回調函數
,將同步操做變成異步操做,f1 不會阻塞程序的運行,f2 也無需空空等待,例如 JQuery 的 ajax。web
回調函數的demo:ajax
function f1(f2){ setTimeout(function(){ console.log('先執行 f1') },1000) f2() } function f2() { console.log('再執行 f2') }
效果以下:編程
總結:回調函數易於實現、便於理解,可是屢次回調會致使代碼高度耦合axios
腳本的執行不取決代碼的順序,而取決於某一個事件是否發生。api
事件監聽的demopromise
$(document).ready(function(){ console.log('DOM 已經 ready') });
發佈/訂閱模式是利用一個消息中心,發佈者發佈一個消息給消息中心,訂閱者從消息中心訂閱該消息,。相似於 vue 的父子組件之間的傳值。app
發佈訂閱模式的 demo
//訂閱done事件 $('#app').on('done',function(data){ console.log(data) }) //發佈事件 $('#app').trigger('done,'haha')
Promise 實際就是一個對象, 從它能夠得到異步操做的消息,Promise 對象有三種狀態,pending(進行中)、fulfilled(已成功)和rejected(已失敗)。Promise 的狀態一旦改變以後,就不會在發生任何變化,將回調函數變成了鏈式調用。
Promise 封裝異步請求demo
export default function getMethods (url){ return new Promise(function(resolve, reject){ axios.get(url).then(res => { resolve(res) }).catch(err =>{ reject(err) }) }) } getMethods('/api/xxx').then(res => { console.log(res) }, err => { console.log(err) })
Generator 函數是一個狀態機,封裝了多個內部狀態。執行 Generator 函數會返回一個遍歷器對象,使用該對象的 next() 方法,能夠遍歷 Generator 函數內部的每個狀態,直到 return 語句。
形式上,Generator 函數是一個普通函數,可是有兩個特徵。一是,function關鍵字與函數名之間有一個星號;二是,函數體內部使用yield表達式, yield是暫停執行的標記。
next() 方法遇到yield表達式,就暫停執行後面的操做,並將緊跟在yield後面的那個表達式的值,做爲返回的對象的value屬性值。
Generator 的 demo
function *generatorDemo() { yield 'hello'; yield 1 + 2; return 'ok'; } var demo = generatorDemo() demo.next() // { value: 'hello', done: false } demo.next() // { value: 3, done: false } demo.next() // { value: 'ok', done: ture } demo.next() // { value: undefined, done: ture }
async函數返回的是一個 Promise 對象,可使用 then 方法添加回調函數,async 函數內部 return 語句返回的值,會成爲 then 方法回調函數的參數。當函數執行的時候,一旦遇到await就會先返回,等到異步操做完成,再接着執行函數體內後面的語句。
1.await命令後面返回的是 Promise 對象,運行結果多是rejected,因此最好把await命令放在try...catch代碼塊中。
async 的 demo1
async function demo() { try { await new Promise(function (resolve, reject) { // something }); } catch (err) { console.log(err); } } demo().then(data => { console.log(data) // })
參考文獻
https://developers.google.com...
http://es6.ruanyifeng.com/