JavaScript異步處理——Generator及async函數(shù)

Generator

Generator 函數(shù)是 ES6 提供的一種異步編程解決方案。Generator函數(shù)在function關(guān)鍵字與函數(shù)名之間有一個(gè)*號(hào)灼芭。

執(zhí)行Generator 函數(shù)返回一個(gè)指向函數(shù)內(nèi)部的指針對(duì)象戚扳,每次調(diào)用該對(duì)象的next方法,指針從函數(shù)頭部或上一次停下來(lái)的地方開(kāi)始執(zhí)行档泽,直到遇到下一個(gè)yield表達(dá)式/return語(yǔ)句攻泼。
next可傳入?yún)?shù),該參數(shù)會(huì)作為上一個(gè)yield表達(dá)式的返回值诡延。
例:

    function* test() {
        yield 'hello';
        const text = yield 'world';
        console.log(text);
        return 'ending';
    }

    let t = test();

    console.log(t.next());//{value: "hello", done: false}
    console.log(t.next());//{value: "world", done: false}                          
    console.log(t.next('!'));//"!"
                            //{value: "ending", done: true}
    console.log(t.next());//{value: undefined, done: true}

假設(shè)有異步方法fetchData滞欠,如果我們希望通過(guò)Generator函數(shù)寫(xiě)出串聯(lián)執(zhí)行的效果:

    function fetchData(data) {
        return function (cb) {
            setTimeout(function () {
                cb(data + 1);
            }, 500);
        }
    }

    function* test() {

        let t1 = yield fetchData(1);
        let t2 = yield fetchData(t1);
    }

    let t = test();
    t.next().value((data)=>{
        t.next(data).value((data)=>{
            t.next(data);
        });
    });

顯然不是我們想要的效果,這時(shí)可以做一個(gè)Generator自動(dòng)運(yùn)行方法肆良,通過(guò)遞歸解決:

    run(test);//這樣一句話就夠了

    function run(gen) {
        const g = gen();
        next();

        function next() {
            const result = g.next(...arguments);
            if (!result.done) {
                result.value(next);
            }
        }
    }

但異步函數(shù)除了使用回調(diào)實(shí)現(xiàn)筛璧,還可以使用Promise實(shí)現(xiàn),這樣我們的自動(dòng)運(yùn)行方法就要加上對(duì)Promise的處理了惹恃。

    function run(gen) {
        const g = gen();
        next();

        function next() {
            const result = g.next(...arguments);
            if (!result.done) {
                const value = result.value;
                
                if(isPromise(value)){
                    value.then(next);
                }else{
                    value(next);
                }
            }
        }

        function isPromise(obj){
            if( obj && typeof obj.then == 'function' ) return true;
            return false;
        }
    }

現(xiàn)在已經(jīng)完成了自動(dòng)執(zhí)行的功能隧哮,但存在兩個(gè)問(wèn)題。

  1. 不能獲取到Generator函數(shù)最終return的值座舍。
  2. 但如果異步操作中發(fā)生了異常沮翔,現(xiàn)在是沒(méi)辦法處理的。

為了解決這兩個(gè)問(wèn)題曲秉,需要作出修改

  • run方法返回一個(gè)Promise采蚀,最終通過(guò)Promise來(lái)返回結(jié)果
  • 遞歸調(diào)用next時(shí),加上onRejected進(jìn)行處理承二。而對(duì)于使用回調(diào)進(jìn)行異步的方法榆鼠,則將其轉(zhuǎn)換成一個(gè)Promise,使代碼可以復(fù)用亥鸠。

如果異步方法返回的不是promise妆够,則包裝成一個(gè)promise來(lái)處理识啦。

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

    function isPromise(obj) {
        if (obj && typeof obj.then === 'function') return true;
        return false;
    }

    function thunkToPromise(obj) {
        return new Promise((resolve, reject) => {
            //error first模式,回調(diào)函數(shù)接收的第一個(gè)參數(shù)為error
            obj(function (err) {
                if (err) return reject(err);
                resolve([].splice.call(arguments, 1));
            });
        })
    }

最后run方法返回一個(gè)promise神妹,這樣外部就可以獲取到自動(dòng)執(zhí)行中遇到的異惩窍或最終結(jié)果了。

    function run(gen) {
        return new Promise((resolve, reject) => {
            const g = gen();
            next();

            function next() {
                const result = g.next(...arguments);
                if (!result.done) {
                    const promise = toPromise(result.value);
                    promise.then(next, reject);//遇到問(wèn)題就用reject返回異常鸵荠,不繼續(xù)遞歸
                } else {
                    resolve(result.value);//返回最終執(zhí)行結(jié)果
                }
            }
        })
    }

async函數(shù)

async函數(shù)是ES2017標(biāo)準(zhǔn)引入的一種異步操作方式冕茅,在function前加上async使用。
例:

    test();
    
    async function test(){
        let t1 = await fetchData('lin');
        console.log(t1);//data of lin
        let t2 = await fetchData('qibin');
        console.log(t2);//data of qibin
    }

    function fetchData(name) {
        return new Promise((resolve, reject) => {
            setTimeout(function () {
                resolve(`data of ${name}`);
            }, 500);
        })
    }

很熟悉有沒(méi)有蛹找!其實(shí)async函數(shù)就是Generator+自動(dòng)執(zhí)行姨伤。
async異步操作超級(jí)優(yōu)雅,是我認(rèn)為js異步操作最好的形式庸疾。

更多關(guān)于Generatorasync函數(shù)

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末乍楚,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子届慈,更是在濱河造成了極大的恐慌徒溪,老刑警劉巖,帶你破解...
    沈念sama閱讀 211,265評(píng)論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件拧篮,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡牵舱,警方通過(guò)查閱死者的電腦和手機(jī)串绩,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,078評(píng)論 2 385
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)芜壁,“玉大人礁凡,你說(shuō)我怎么就攤上這事』弁” “怎么了顷牌?”我有些...
    開(kāi)封第一講書(shū)人閱讀 156,852評(píng)論 0 347
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)塞淹。 經(jīng)常有香客問(wèn)我窟蓝,道長(zhǎng),這世上最難降的妖魔是什么饱普? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 56,408評(píng)論 1 283
  • 正文 為了忘掉前任运挫,我火速辦了婚禮,結(jié)果婚禮上套耕,老公的妹妹穿的比我還像新娘谁帕。我一直安慰自己,他們只是感情好冯袍,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,445評(píng)論 5 384
  • 文/花漫 我一把揭開(kāi)白布匈挖。 她就那樣靜靜地躺著碾牌,像睡著了一般。 火紅的嫁衣襯著肌膚如雪儡循。 梳的紋絲不亂的頭發(fā)上舶吗,一...
    開(kāi)封第一講書(shū)人閱讀 49,772評(píng)論 1 290
  • 那天,我揣著相機(jī)與錄音贮折,去河邊找鬼裤翩。 笑死,一個(gè)胖子當(dāng)著我的面吹牛调榄,可吹牛的內(nèi)容都是我干的踊赠。 我是一名探鬼主播,決...
    沈念sama閱讀 38,921評(píng)論 3 406
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼每庆,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼筐带!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起缤灵,我...
    開(kāi)封第一講書(shū)人閱讀 37,688評(píng)論 0 266
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤伦籍,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后腮出,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體帖鸦,經(jīng)...
    沈念sama閱讀 44,130評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,467評(píng)論 2 325
  • 正文 我和宋清朗相戀三年胚嘲,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了作儿。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,617評(píng)論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡馋劈,死狀恐怖攻锰,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情妓雾,我是刑警寧澤娶吞,帶...
    沈念sama閱讀 34,276評(píng)論 4 329
  • 正文 年R本政府宣布,位于F島的核電站械姻,受9級(jí)特大地震影響妒蛇,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜楷拳,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,882評(píng)論 3 312
  • 文/蒙蒙 一材部、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧唯竹,春花似錦乐导、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,740評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)旺拉。三九已至,卻和暖如春棵磷,著一層夾襖步出監(jiān)牢的瞬間蛾狗,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 31,967評(píng)論 1 265
  • 我被黑心中介騙來(lái)泰國(guó)打工仪媒, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留沉桌,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 46,315評(píng)論 2 360
  • 正文 我出身青樓算吩,卻偏偏與公主長(zhǎng)得像留凭,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子偎巢,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,486評(píng)論 2 348

推薦閱讀更多精彩內(nèi)容