JavaScript 異步調(diào)用框架 (Part 2 - 用例設(shè)計(jì))
更新時(shí)間:2009年08月03日 23:53:28 作者:
在上一篇文章里說(shuō)到,我們要設(shè)計(jì)一個(gè)異步調(diào)用框架,最好能夠統(tǒng)一同步異步調(diào)用的接口,同時(shí)具體調(diào)用順序與實(shí)現(xiàn)方式無(wú)關(guān)。那么我們現(xiàn)在就來(lái)設(shè)計(jì)這樣一個(gè)框架的用例。
傳遞回調(diào)
我們首先要考慮的一個(gè)問(wèn)題是,如何傳遞回調(diào)入口。在最傳統(tǒng)的XHR調(diào)用當(dāng)中,回調(diào)函數(shù)會(huì)被作為最后一個(gè)參數(shù)傳遞給異步函數(shù):
function asyncOperation(argument, callback)
在參數(shù)相當(dāng)多的時(shí)候,我們可以把參數(shù)放到一個(gè)JSON里面,這樣參數(shù)就如同具名參數(shù)一樣,可以通過(guò)參數(shù)名選擇性的傳遞參數(shù),不傳遞的參數(shù)相當(dāng)于使用默認(rèn)值。這是從Prototype開始就流行起來(lái)的做法:
function asyncOperation(argument, options)
然而這兩種做法都有一個(gè)壞處,就是把同步函數(shù)改為異步函數(shù)(或同步異步混合函數(shù))時(shí),必須顯式地修改函數(shù)簽名,在最后增加一個(gè)(或多個(gè))參數(shù)。
由于在調(diào)用棧的底層引入異步函數(shù)對(duì)我們來(lái)說(shuō)太常見了,為此可能要更改一大堆上層調(diào)用函數(shù)簽名的成本實(shí)在是太高了,所以我們還是想一個(gè)不用修改函數(shù)簽名的做法吧。
在這里我參考了.NET Framework的IAsyncResult設(shè)計(jì),把異步操作有關(guān)的一切信息集中到一個(gè)對(duì)象上來(lái),從而避免了對(duì)函數(shù)簽名的修改。在此,我們假設(shè)一個(gè)異步函數(shù)的調(diào)用原型是這樣子的:
function asyncOperation(argument) {
operation = new Async.Operation();
setTimeout(function() { operation.yield("hello world"); }, 1000);
return operation;
}
在這段代碼里,我們返回了一個(gè)Operation對(duì)象,用于將來(lái)傳遞回調(diào)函數(shù)。同時(shí),我們通過(guò)setTimeout模擬了異步返回結(jié)果,而具體的返回方式就是yield方法。
接著,我們還要設(shè)計(jì)傳遞回調(diào)函數(shù)的方法。由于我們不能好像C#那樣重載+=運(yùn)算符,所以只能用函數(shù)傳遞回調(diào)函數(shù):
var operation = asyncOperation(argument);
operation.addCallback(function(result) { alert(result); });
在C#里面做這樣的設(shè)計(jì)是不安全的,因?yàn)樵诋惒讲僮骺赡茉谔砑踊卣{(diào)之前就完成了。但在JavaScript里面這樣寫是安全的,因?yàn)镴avaScript是單線程的,緊接著asyncOperation的同步addCallback必然先執(zhí)行,asyncOperation中的異步y(tǒng)ield必然后執(zhí)行。
調(diào)用順序
可能有人要問(wèn),如果用戶使用同步的方式來(lái)調(diào)用yield,這時(shí)候執(zhí)行順序不一樣依賴于yield的實(shí)現(xiàn)嗎?沒錯(cuò),不過(guò)yeild是在框架中一次性實(shí)現(xiàn)的,我們只要把它做成異步的就可以了,這樣即使對(duì)它進(jìn)行同步調(diào)用,也不影響執(zhí)行順序:
function psudoAsyncOperation(argument) {
operation = new Async.Operation();
operation.yield("hello world");
return operation;
}
var operation = asyncOperation(argument);
operation.addCallback(function(result) { alert(result); });
就算把代碼寫成這個(gè)樣子,我們也能確保addCallback先于yield的實(shí)際邏輯執(zhí)行。
事后回調(diào)
有時(shí)候,框架的使用者可能真的寫出了先yield后addCallback的代碼。這時(shí)候,我認(rèn)為必須保證addCallback中添加的回調(diào)函數(shù)會(huì)被立即觸發(fā)。因?yàn)橛脩籼砑舆@個(gè)回調(diào)函數(shù),意味著他期望當(dāng)異步操作有結(jié)果時(shí)通知這個(gè)回調(diào)函數(shù),而這與添加回調(diào)函數(shù)時(shí)異步操作是否完成無(wú)關(guān)。為此,我們?cè)偬砑右粋€(gè)用例:
function psudoAsyncOperation(argument) {
operation = new Async.Operation();
operation.yield("hello world");
return operation;
}
var operation = asyncOperation(argument);
setTimeout(function() {
operation.addCallback(function(result) { alert(result); });
}, 1000);
小結(jié)
到這里,我們就設(shè)計(jì)好了一個(gè)名為Async.Operation的異步操作對(duì)象,具體如何實(shí)現(xiàn)關(guān)鍵的yield方法和addCallback方法將在下一篇文章講述如果。
我們首先要考慮的一個(gè)問(wèn)題是,如何傳遞回調(diào)入口。在最傳統(tǒng)的XHR調(diào)用當(dāng)中,回調(diào)函數(shù)會(huì)被作為最后一個(gè)參數(shù)傳遞給異步函數(shù):
復(fù)制代碼 代碼如下:
function asyncOperation(argument, callback)
在參數(shù)相當(dāng)多的時(shí)候,我們可以把參數(shù)放到一個(gè)JSON里面,這樣參數(shù)就如同具名參數(shù)一樣,可以通過(guò)參數(shù)名選擇性的傳遞參數(shù),不傳遞的參數(shù)相當(dāng)于使用默認(rèn)值。這是從Prototype開始就流行起來(lái)的做法:
復(fù)制代碼 代碼如下:
function asyncOperation(argument, options)
然而這兩種做法都有一個(gè)壞處,就是把同步函數(shù)改為異步函數(shù)(或同步異步混合函數(shù))時(shí),必須顯式地修改函數(shù)簽名,在最后增加一個(gè)(或多個(gè))參數(shù)。
由于在調(diào)用棧的底層引入異步函數(shù)對(duì)我們來(lái)說(shuō)太常見了,為此可能要更改一大堆上層調(diào)用函數(shù)簽名的成本實(shí)在是太高了,所以我們還是想一個(gè)不用修改函數(shù)簽名的做法吧。
在這里我參考了.NET Framework的IAsyncResult設(shè)計(jì),把異步操作有關(guān)的一切信息集中到一個(gè)對(duì)象上來(lái),從而避免了對(duì)函數(shù)簽名的修改。在此,我們假設(shè)一個(gè)異步函數(shù)的調(diào)用原型是這樣子的:
復(fù)制代碼 代碼如下:
function asyncOperation(argument) {
operation = new Async.Operation();
setTimeout(function() { operation.yield("hello world"); }, 1000);
return operation;
}
在這段代碼里,我們返回了一個(gè)Operation對(duì)象,用于將來(lái)傳遞回調(diào)函數(shù)。同時(shí),我們通過(guò)setTimeout模擬了異步返回結(jié)果,而具體的返回方式就是yield方法。
接著,我們還要設(shè)計(jì)傳遞回調(diào)函數(shù)的方法。由于我們不能好像C#那樣重載+=運(yùn)算符,所以只能用函數(shù)傳遞回調(diào)函數(shù):
復(fù)制代碼 代碼如下:
var operation = asyncOperation(argument);
operation.addCallback(function(result) { alert(result); });
在C#里面做這樣的設(shè)計(jì)是不安全的,因?yàn)樵诋惒讲僮骺赡茉谔砑踊卣{(diào)之前就完成了。但在JavaScript里面這樣寫是安全的,因?yàn)镴avaScript是單線程的,緊接著asyncOperation的同步addCallback必然先執(zhí)行,asyncOperation中的異步y(tǒng)ield必然后執(zhí)行。
調(diào)用順序
可能有人要問(wèn),如果用戶使用同步的方式來(lái)調(diào)用yield,這時(shí)候執(zhí)行順序不一樣依賴于yield的實(shí)現(xiàn)嗎?沒錯(cuò),不過(guò)yeild是在框架中一次性實(shí)現(xiàn)的,我們只要把它做成異步的就可以了,這樣即使對(duì)它進(jìn)行同步調(diào)用,也不影響執(zhí)行順序:
復(fù)制代碼 代碼如下:
function psudoAsyncOperation(argument) {
operation = new Async.Operation();
operation.yield("hello world");
return operation;
}
var operation = asyncOperation(argument);
operation.addCallback(function(result) { alert(result); });
就算把代碼寫成這個(gè)樣子,我們也能確保addCallback先于yield的實(shí)際邏輯執(zhí)行。
事后回調(diào)
有時(shí)候,框架的使用者可能真的寫出了先yield后addCallback的代碼。這時(shí)候,我認(rèn)為必須保證addCallback中添加的回調(diào)函數(shù)會(huì)被立即觸發(fā)。因?yàn)橛脩籼砑舆@個(gè)回調(diào)函數(shù),意味著他期望當(dāng)異步操作有結(jié)果時(shí)通知這個(gè)回調(diào)函數(shù),而這與添加回調(diào)函數(shù)時(shí)異步操作是否完成無(wú)關(guān)。為此,我們?cè)偬砑右粋€(gè)用例:
復(fù)制代碼 代碼如下:
function psudoAsyncOperation(argument) {
operation = new Async.Operation();
operation.yield("hello world");
return operation;
}
var operation = asyncOperation(argument);
setTimeout(function() {
operation.addCallback(function(result) { alert(result); });
}, 1000);
小結(jié)
到這里,我們就設(shè)計(jì)好了一個(gè)名為Async.Operation的異步操作對(duì)象,具體如何實(shí)現(xiàn)關(guān)鍵的yield方法和addCallback方法將在下一篇文章講述如果。
您可能感興趣的文章:
- JavaScript 異步調(diào)用框架 (Part 1 - 問(wèn)題 & 場(chǎng)景)
- JavaScript 異步調(diào)用框架 (Part 3 - 代碼實(shí)現(xiàn))
- JavaScript 異步調(diào)用框架 (Part 4 - 鏈?zhǔn)秸{(diào)用)
- JavaScript 異步調(diào)用框架 (Part 5 - 鏈?zhǔn)綄?shí)現(xiàn))
- javascript 支持鏈?zhǔn)秸{(diào)用的異步調(diào)用框架Async.Operation
- JavaScript 異步調(diào)用框架 (Part 6 - 實(shí)例 & 模式)
- JavaScript異步調(diào)用定時(shí)方法并停止該方法實(shí)現(xiàn)代碼
- Java多線程實(shí)現(xiàn)異步調(diào)用的方法
相關(guān)文章
Webwork 實(shí)現(xiàn)文件上傳下載代碼詳解
WebWork 當(dāng)然也提供了很友好的攔截器來(lái)實(shí)現(xiàn)對(duì)文件的上傳,讓我們可以專注與業(yè)務(wù)邏輯的設(shè)計(jì)和實(shí)現(xiàn),在實(shí)現(xiàn)上傳和下載時(shí)順便關(guān)注了下框架上傳下載的實(shí)現(xiàn)2016-02-02bootstrapValidator表單驗(yàn)證插件學(xué)習(xí)
這篇文章主要為大家詳細(xì)介紹了表單驗(yàn)證插件bootstrapValidator的使用方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-12-12JavaScript實(shí)現(xiàn)各種排序的代碼詳解
這篇文章給大家介紹了js實(shí)現(xiàn)各種排序功能,包括冒泡排序,選擇排序和插入排序,代碼簡(jiǎn)單易懂,非常不錯(cuò),具有參考借鑒價(jià)值,需要的的朋友參考下吧2017-08-08探索JavaScript函數(shù)的無(wú)限可能(函數(shù)基本概念)
JavaScript中的函數(shù)是一種重要的編程概念,它允許我們封裝可重用的代碼塊,并在需要時(shí)進(jìn)行調(diào)用,本文將深入介紹JavaScript函數(shù)的各個(gè)方面,包括函數(shù)定義和調(diào)用、參數(shù)和返回值、作用域和閉包、高階函數(shù)以及常見的函數(shù)應(yīng)用場(chǎng)景,感興趣的朋友一起看看吧2023-08-08JS把內(nèi)容動(dòng)態(tài)插入到DIV的實(shí)現(xiàn)方法
下面小編就為大家?guī)?lái)一篇JS把內(nèi)容動(dòng)態(tài)插入到DIV的實(shí)現(xiàn)方法。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2016-07-07