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

JavaScript 異步調用框架 (Part 4 - 鏈式調用)

現實開發中,要按順序執行一系列的同步異步操作又是很常見的。還是用百度Hi網頁版中的例子,我們先要異步獲取聯系人列表,然后再異步獲取每一個聯系人的具體信息,而且后者是分頁獲取的,每次請求發送10個聯系人的名稱然后取回對應的具體信息。這就是多個需要順序執行的異步請求。
為此,我們需要設計一種新的操作方式來優化代碼可讀性,讓順序異步操作代碼看起來和傳統的順序同步操作代碼一樣優雅。
傳統做法
大多數程序員都能夠很好的理解順序執行的代碼,例如這樣子的:
復制代碼 代碼如下:
var firstResult = firstOperation(initialArgument);
var secondResult = secondOperation(firstResult);
var finalResult = thirdOperation(secondResult);
alert(finalResult);

其中先執行的函數為后執行的函數提供所需的數據。然而使用我們的異步調用框架后,同樣的邏輯必須變成這樣子:
復制代碼 代碼如下:
firstAsyncOperation(initialArgument).addCallback(function(firstResult) {
secondAsyncOperation(firstResult).addCallback(function(secondResult) {
thirdAsyncOperation(secondResult).addCallback(function(finalResult) {
alert(finalResult);
});
});
});

鏈式寫法
我認為上面的代碼實在是太不美觀了,并且希望能夠改造為jQuery風格的鏈式寫法。為此,我們先構造一個用例:
復制代碼 代碼如下:
Async.go(initialArgument)
.next(firstAsyncOperation)
.next(secondAsyncOperation)
.next(thirdAsyncOperation)
.next(function(finalResult) { alert(finalResult); })

在這個用例當中,我們在go傳入初始化數據,然后每一個next后面傳入一個數據處理函數,這些處理函數按順序對數據進行處理。
同步并存
上面的用例調用到的全部都是異步函數,不過我們最好能夠兼容同步函數,讓使用者無需關心函數的具體實現,也能使用這項功能。為此我們再寫一個這樣的用例:
復制代碼 代碼如下:
Async.go(0)
.next(function(i) { alert(i); return i + 1; })
.next(function(i) {
alert(i);
var operation = new Async.Operation();
setTimeout(function() { operation.yield(i + 1); }, 1000);
return operation;
})
.next(function(i) { alert(i); return i + 1; })
.next(function(i) { alert(i); return i; });

在上述用例中,我們期待能夠看到0, 1, 2, 3的提示信息序列,并且1和2之間間隔為1000毫秒。
異步本質
一個鏈式調用,本質上也是一個異步調用,所以它返回的也是一個Operation實例。這個實例自然也有result、state和completed這幾個字段,并且當整個鏈式調用完成時,result等于最后一個調用返回的結果,而completed自然是等于true。
我們可以擴展一下上一個用例,得到如下用例代碼:
復制代碼 代碼如下:
var chainOperation = Async.go(0)
.next(function(i) { alert(i); return i + 1; })
.next(function(i) {
alert(i);
var operation = new Async.Operation();
setTimeout(function() { operation.yield(i + 1); }, 1000);
return operation;
})
.next(function(i) { alert(i); return i + 1; })
.next(function(i) { alert(i); return i; });
setTiemout(function() { alert(chainOperation.result; }, 2000);

把鏈式調用的返回保存下來,在鏈式調用完成時,它的result應該與最后一個操作的返回一致。在上述用例中,也就是3。
調用時機
盡管我們提供了一種鏈式調用方式,但是用戶不一定會按照這種固定的方式來調用,所以我們仍然要考慮兼容用戶的各種可能用法,例如說異步地用next往調用鏈添加操作:
復制代碼 代碼如下:
var chainOperation = Async.go(0);
chainOperation.next(function(i) { alert(i); return i + 1; });
setTimeout(function() {
chainOperation.next(function(i) {
alert(i);
var operation = new Async.Operation();
setTimeout(function() { operation.yield(i + 1); }, 2000);
return operation;
})
}, 1000);
setTimeout(function() {
chainOperation.next(function(i) { alert(i); return i + 1; });
}, 2000);

在這個用例當中,用戶每隔1000毫秒添加一個操作,而其中第二個操作耗時2000毫秒。也就是說,添加第三個操作時第二個操作還沒返回。作為一個健壯的框架,必須要能兼容這樣的使用方式。
此外我們還要考慮,用戶可能想要先構造調用鏈,然后再執行調用鏈。這時候用戶就會先使用next方法添加操作,再使用go方法執行。
復制代碼 代碼如下:
var chainOperation = Async
.chain(function(i) { alert(i); return i + 1; })
.next(function(i) {
alert(i);
var operation = new Async.Operation();
setTimeout(function() { operation.yield(i + 1); }, 2000);
return operation;
})
.go(0)
setTimeout(function() {
chainOperation.next(function(i) { alert(i); return i + 1; })
}, 1000);

在上述用例中,用戶通過chain和next添加了頭同步操作和異步操作各一個,然后用go執行調用鏈,在調用鏈執行完畢之前又用next異步追加了一個操作。一個健壯的框架,在這樣的用例當中應該能夠如同用戶所期望的那樣提示0, 1, 2。
小結
針對鏈式調用的需求,我們設計了如此多的用例,包括各種奇怪的異步調用方式。最終如何實現這樣的功能呢?

JavaScript技術JavaScript 異步調用框架 (Part 4 - 鏈式調用),轉載需保留來源!

鄭重聲明:本文版權歸原作者所有,轉載文章僅為傳播更多信息之目的,如作者信息標記有誤,請第一時間聯系我們修改或刪除,多謝。

主站蜘蛛池模板: 拍床戏被肉高H纯肉H在水 | 亚洲永久精品ww47 | 国产Av影片麻豆精品传媒 | 国产精品1卡二卡三卡四卡乱码 | 成人在免费观看视频国产 | 欧美 国产 日产 韩国 在线 | av女优快播 | 伊人久久大香线蕉avapp下载 | 精品精品国产自在现拍 | 伦理片97影视网 | 高跟丝袜岳第一次 | 男男gaygay拳头 | 国产AV亚洲精品久久久久软件 | 综合网伊人 | 人人草人人草 | 嘟嘟嘟影院免费观看视频 | 岛国片免费在线观看 | 最近中文字幕MV高清在线视频 | 老熟女重囗味GRANNYBBW | 亚洲破处女 | 影音先锋av男人资源 | 同时被两个男人轮流舔 | 美女张开腿让男人桶爽无弹窗 | 欧美亚洲色帝国 | 免费观看成人毛片 | 99久久精品免费精品国产 | 青青热久精品国产亚洲AV无码 | 九九九色成人网 | 97视频在线播放 | AV无码国产精品午夜A片麻豆 | 午夜日本大胆裸艺术 | 97国产成人精品视频 | 美女全光末满18勿进 | 亚洲 日韩 国产 制服 在线 | 亚洲一区在线观看无码欧美 | 翘臀后进美女白嫩屁股视频 | 亚洲国产区中文在线观看 | 亚洲日韩天堂在线中文字幕 | 一个人免费观看HD完整版 | 久久艹伊人 | 97精品伊人久久大香线蕉app |