天天躁日日躁狠狠躁AV麻豆-天天躁人人躁人人躁狂躁-天天澡夜夜澡人人澡-天天影视香色欲综合网-国产成人女人在线视频观看-国产成人女人视频在线观看

JavaScript 異步調(diào)用框架 (Part 1 - 問題 & 場(chǎng)景)

問題
在Ajax應(yīng)用中,調(diào)用XMLHttpRequest是很常見的情況。特別是以客戶端為中心的Ajax應(yīng)用,各種需要從服務(wù)器端獲取數(shù)據(jù)的操作都通過(guò)XHR異步調(diào)用完成。然而在單線程的JavaScript編程中,XHR異步調(diào)用的代碼風(fēng)格實(shí)在是與一般的JavaScript代碼格格不入。

額外參數(shù)
考慮一個(gè)除法函數(shù),如果它是純客戶端的同步函數(shù),那么簽名會(huì)是這樣的:

function divide(operand1, operand2)

然而假設(shè)我們對(duì)客戶端除法的精度不滿意,于是把除法轉(zhuǎn)移到服務(wù)器端來(lái)執(zhí)行,那么它是個(gè)需要調(diào)用XHR的異步函數(shù),簽名也就可能會(huì)是以下幾種之一:
復(fù)制代碼 代碼如下:
function divide(operand1, operand2, callback)
function divide(operand1, operand2, successCallback, failureCallback)
function divide(operand1, operand2, options)


我們必須在簽名中引入新的參數(shù)來(lái)傳遞回調(diào)函數(shù),不能選擇讓函數(shù)變成阻塞式的同步調(diào)用。

可傳遞性
不僅僅直接操作XHR的函數(shù)需要引入新的參數(shù),這種復(fù)雜性還會(huì)順著調(diào)用棧向外傳遞。例如說(shuō),我們對(duì)加減乘除四則運(yùn)算作了封裝,只向外暴露一個(gè)運(yùn)算接口:

function calculate(operand1, operand2, operator)

這個(gè)calculate函數(shù)根據(jù)operator參數(shù)來(lái)調(diào)用內(nèi)部的plus, subtract, multiply, divide函數(shù)。然而,因?yàn)閐ivide函數(shù)變成了異步函數(shù),所以整個(gè)calculate函數(shù)不得不也轉(zhuǎn)變?yōu)楫惒胶瘮?shù):

function calculate(operand1, operand2, operator, callback)

同時(shí),在調(diào)用棧之上凡是需要調(diào)用到calculate的函數(shù),都必須變成異步的,除非它并不需要向上一級(jí)調(diào)用函數(shù)返回結(jié)果。

同步并存
盡管calculate函數(shù)變成了一個(gè)異步函數(shù),它所調(diào)用的plus, subtract, multiply函數(shù)還是同步函數(shù),只有調(diào)用divide時(shí)是異步的,這時(shí)候calculate就是一個(gè)異步同步并存函數(shù)。

這會(huì)帶來(lái)什么問題?calculate的調(diào)用者看到函數(shù)簽名自然會(huì)認(rèn)為calculate是個(gè)異步函數(shù),因?yàn)樗枰獋鬟f回調(diào)函數(shù),然而calculate的執(zhí)行方式卻是不確定的。考慮如下調(diào)用:

calculate(operand1, operand2, operator, callback);
next();

這里涉及到callback和next兩個(gè)函數(shù),它們哪個(gè)先執(zhí)行哪個(gè)后執(zhí)行是不確定的,或者說(shuō)是依賴于calculate具體實(shí)現(xiàn)的。

如果calculate的實(shí)現(xiàn)是,當(dāng)不需要進(jìn)行異步操作時(shí),直接調(diào)用callback。那么,在執(zhí)行加減乘法時(shí)callback會(huì)在next之前被調(diào)用;在執(zhí)行除法時(shí)next會(huì)在callback之前調(diào)用。

如果我們不喜歡這種不確定性,我們可以改變一下calculate的實(shí)現(xiàn),把同步調(diào)用也都改為setTimeout形式的,這樣在任何情況下next都一定會(huì)在callback之前被調(diào)用。

然而后面一種做法依賴于成本較高的實(shí)現(xiàn)方式,開發(fā)者一個(gè)不小心(或者擺明偷懶)就會(huì)漏掉setTimeout,導(dǎo)致函數(shù)調(diào)用順序變得不確定,所以我們會(huì)希望這是框架幫助實(shí)現(xiàn)的功能,在使用框架時(shí)無(wú)法把這繞過(guò)。

場(chǎng)景
在這里,我舉一個(gè)關(guān)于上述問題的具體應(yīng)用場(chǎng)景。(為簡(jiǎn)化問題,描述已略作修改,與實(shí)際應(yīng)用并不一致。)

在百度Hi網(wǎng)頁(yè)版里面,我們會(huì)在客戶端保存一個(gè)用戶對(duì)象列表,在打開和這個(gè)用戶的聊天窗口時(shí),我們需要從中讀取這個(gè)用戶的信息。這個(gè)操作就涉及很多可能同步又可能異步的分支:

用戶對(duì)象未緩存
異步讀取用戶信息
用戶對(duì)象已緩存
用戶是好友(信息更新會(huì)由服務(wù)器端推送)
同步讀取用戶信息
用戶不是好友(信息更新需要由客戶端拉取)
可以接受緩存信息
同步讀取用戶信息
必須獲取最新信息
異步讀取用戶信息
可以看到,分支的結(jié)果最終既有同步的,也有異步的。并且這些分支還不是在一個(gè)函數(shù)里完成,而是在幾個(gè)函數(shù)里實(shí)現(xiàn)。也就是說(shuō),按照傳統(tǒng)的模式,這些函數(shù)一部分是同步的,一部分是異步的,由于異步的傳遞性,最終調(diào)用棧頂層的函數(shù)都是異步的。

為了解決這個(gè)問題,我們需要寫一個(gè)異步調(diào)用框架,用一種統(tǒng)一的方式來(lái)進(jìn)行調(diào)用,把同步和異步調(diào)用都合并為一種返回方式。

JavaScript技術(shù)JavaScript 異步調(diào)用框架 (Part 1 - 問題 & 場(chǎng)景),轉(zhuǎn)載需保留來(lái)源!

鄭重聲明:本文版權(quán)歸原作者所有,轉(zhuǎn)載文章僅為傳播更多信息之目的,如作者信息標(biāo)記有誤,請(qǐng)第一時(shí)間聯(lián)系我們修改或刪除,多謝。

主站蜘蛛池模板: 美女拔萝卜 | 亚洲伊人久久网 | freevideoshd | 久久国产欧美日韩精品免费 | 正在播放国产精品 | 蜜桃传媒在线观看 | 在线视频免费国产成人 | 风月宝鉴之淫乱英雄传 电影 | 日本一本道高清码v | 国产欧美一本道无码 | 免费国产午夜理论不卡 | 久久视频这里只精品99热在线观看 | 国产人妻系列无码专区97SS | av影音先锋影院男人站 | 久久亚洲AV成人无码动态图 | 野花视频在线观看免费 | 亚洲国产精品99久久久久久 | 某上海少妇3P黑人完整版BD | 国产精品成人免费视频99 | 教室眠催白丝美女校花 | 嗯啊插的好深啊使劲抽视频 | 好紧小嫩嫩水的10p 好紧好湿太硬了我太爽了小说 | 亚洲欧美一区二区三区蜜芽 | 男生插女生下体 | 成人永久免费视频 | 俄罗斯性孕妇孕交 | 熟女少妇内射日韩亚洲 | 青草影院内射中出高潮-百度 | 男男高H啪肉Np文多攻多一受 | 国内精品乱码卡一卡2卡三卡新区 | 亚洲成人免费看 | 日本中文一区 | 亚洲日本欧美国产在线视 | 日本阿v直播在线 | 色婷婷狠狠97成为人免费 | 阿娇和冠希13分钟在线观看 | 亚洲 国产 日韩 欧美 在线 | 久久国产36精品色熟妇 | 国内精品七七久久影院 | 特黄特色大片免费播放器9 特黄特黄aaaa级毛片免费看 | 男生jj插入女生jj |