四虎精品视频-四虎精品成人免费网站-四虎黄色网-四虎国产视频-国产免费91-国产蜜臀97一区二区三区

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 - 鏈式調用),轉載需保留來源!

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

主站蜘蛛池模板: 胖女人做爰全过程免费看视频| 远方的故乡简谱| 秋天不回来吉他谱扫弦| 创业史全文阅读| 女生的宿舍2| 赤月| dota2反和谐| 消失的客人| 朱茵拍过的三级的电影| 尹雪喜演的全部电影| 小野惠令奈| 刘乐| 赵大鹏简介| 砌体工程质量验收规范gb50203---2011| 韩国电影《我是谁》演员表介绍| 麻豆av视频| 红灯停绿灯行电影观看| 杯弓蛇影读后感| 刑道荣| 刘海东| 健康中国科普先行直播回放| 《杨贵妃淫史》三级| 355 电影| 劳力士电话客服电话24小时| 洞房艳谈| 以下关于宏病毒说法正确的是| kaori全部av作品大全| 音乐会电视剧免费观看完整版| 梦想建筑师泰国百合剧| 拆迁补偿合同| 请假单| 琅琊榜 豆瓣| 我们的祖国是花园简谱| 婴儿什么时候添加辅食最好| 意大利a级情欲片女人城| 吻激情| 邓为个人资料简介图片| 秘密之门| 解决问题五上数学| 日韩电影免费观看高清完整版在线| 方言的战争在线观看高清免费完整版|