国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

探索Javascript異步編程

開發(fā) 前端
異步編程帶來的問題在客戶端Javascript中并不明顯,但隨著服務器端Javascript越來越廣的被使用,大量的異步IO操作使得該問題變得明顯。許多不同的方法都可以解決這個問題,本文討論了一些方法,但并不深入。大家需要根據(jù)自己的情況選擇一個適于自己的方法。

異步編程帶來的問題在客戶端Javascript中并不明顯,但隨著服務器端Javascript越來越廣的被使用,大量的異步IO操作使得該問題變得明顯。許多不同的方法都可以解決這個問題,本文討論了一些方法,但并不深入。大家需要根據(jù)自己的情況選擇一個適于自己的方法。

探索Javascript異步編程

 

筆者在之前的一片博客中簡單的討論了Python和Javascript的異同,其實作為一種編程語言Javascript的異步編程是一個非常值得討論的有趣話題。

JavaScript 異步編程簡介

回調(diào)函數(shù)和異步執(zhí)行

所謂的異步指的是函數(shù)的調(diào)用并不直接返回執(zhí)行的結(jié)果,而往往是通過回調(diào)函數(shù)異步的執(zhí)行。

我們先看看回調(diào)函數(shù)是什么:

  1. var fn = function(callback) { 
  2.     // do something here 
  3.     ... 
  4.     callback.apply(this, para); 
  5. }; 
  6.    
  7. var mycallback = function(parameter) { 
  8.     // do someting in customer callback 
  9. }; 
  10.    
  11. // call the fn with callback as parameter 
  12. fn(mycallback); 

回調(diào)函數(shù),其實就是調(diào)用用戶提供的函數(shù),該函數(shù)往往是以參數(shù)的形式提供的。回調(diào)函數(shù)并不一定是異步執(zhí)行的。比如上述的例子中,回調(diào)函數(shù)是被同步執(zhí)行的。大部分語言都支持回調(diào),C++可用通過函數(shù)指針或者回調(diào)對象,Java一般也是使用回調(diào)對象。

在Javascript中有很多通過回調(diào)函數(shù)來執(zhí)行的異步調(diào)用,例如setTimeout()或者setInterval()。

  1. setTimeout(function(){ 
  2.     console.log("this will be exectued after 1 second!"); 
  3. },1000); 

在 以上的例子中,setTimeout直接返回,匿名函數(shù)會在1000毫秒(不一定能保證是1000毫秒)后異步觸發(fā)并執(zhí)行,完成打印控制臺的操作。也就是 說在異步操作的情境下,函數(shù)直接返回,把控制權(quán)交給回調(diào)函數(shù),回調(diào)函數(shù)會在以后的某一個時間片被調(diào)度執(zhí)行。那么為什么需要異步呢?為什么不能直接在當前函 數(shù)中完成操作呢?這就需要了解Javascript的線程模型了。

#p#

Javascript線程模型和事件驅(qū)動

Javascript 最初是被設計成在瀏覽器中輔助提供HTML的交互功能。在瀏覽器中都包含一個Javascript引擎,Javscript程序就運行在這個引擎之中,并 且只有一個線程。單線程能都帶來很多優(yōu)點,程序員們可以很開心的不用去考慮諸如資源同步,死鎖等多線程阻塞式編程所需要面對的惱人的問題。但是很多人會 問,既然Javascript是單線程的,那它又如何能夠異步的執(zhí)行呢?

這 就需要了解到Javascript在瀏覽器中的事件驅(qū)動(event driven)機制。事件驅(qū)動一般通過事件循環(huán)(event loop)和事件隊列(event queue)來實現(xiàn)的。假定瀏覽器中有一個專門用于事件調(diào)度的實例(該實例可以是一個線程,我們可以稱之為事件分發(fā)線程event dispatch thread),該實例的工作就是一個不結(jié)束的循環(huán),從事件隊列中取出事件,處理所有很事件關聯(lián)的回調(diào)函數(shù)(event handler)。注意回調(diào)函數(shù)是在Javascript的主線程中運行的,而非事件分發(fā)線程中,以保證事件處理不會發(fā)生阻塞。

Event Loop Code:

  1. while(true) { 
  2.  var event = eventQueue.pop(); 
  3.  if(event && event.handler) { 
  4.      event.handler.execute(); // execute the callback in Javascript thread 
  5.  } else { 
  6.      sleep(); //sleep some time to release the CPU do other stuff 
  7.  } 

通過事件驅(qū)動機制,我們可以想象Javascript的編程模型就是響應一系列的事件,執(zhí)行對應的回調(diào)函數(shù)。很多UI框架都采用這樣的模型(例如Java Swing)。

那為什要異步呢,同步不是很好么?

異步的主要目的是處理非阻塞,在和HTML交互的過程中,會需要一些IO操作(典型的就是Ajax請求,腳本文件加載),如果這些操作是同步的,就會阻塞其它操作,用戶的體驗就是頁面失去了響應。

綜上所述Javascript通過事件驅(qū)動機制,在單線程模型下,以異步回調(diào)函數(shù)的形式來實現(xiàn)非阻塞的IO操作。

Javascript異步編程帶來的挑戰(zhàn)

Javascript的單線程模型有很多好處,但同時也帶來了很多挑戰(zhàn)。

代碼可讀性

想象一下,如果某個操作需要經(jīng)過多個非阻塞的IO操作,每一個結(jié)果都是通過回調(diào),程序有可能會看上去像這個樣子。

  1. operation1(function(err, result) { 
  2.     operation2(function(err, result) { 
  3.         operation3(function(err, result) { 
  4.             operation4(function(err, result) { 
  5.                 operation5(function(err, result) { 
  6.                     // do something useful 
  7.                 }) 
  8.             }) 
  9.         }) 
  10.     }) 
  11. }) 

我們稱之為意大利面條式(spaghetti)的代碼。這樣的代碼很難維護。這樣的情況更多的會發(fā)生在server side的情況下。

流程控制

異步帶來的另一個問題是流程控制,舉個例子,我要訪問三個網(wǎng)站的內(nèi)容,當三個網(wǎng)站的內(nèi)容都得到后,合并處理,然后發(fā)給后臺。代碼可以這樣寫:

  1. var urls = ['url1','url2','url3']; 
  2. var result = []; 
  3.    
  4. for (var i = 0, len = urls.length(); i < len; i++ ) { 
  5.     $.ajax({ 
  6.         url: urls[i], 
  7.         context: document.body, 
  8.         success: function(){ 
  9.           //do something on success 
  10.           result.push("one of the request done successfully"); 
  11.           if (result.length === urls.length()) { 
  12.               //do something when all the request is completed successfully 
  13.           } 
  14.         }}); 

上述代碼通過檢查result的長度的方式來決定是否所有的請求都處理完成,這是一個很丑陋方法,也很不可靠。

異常和錯誤處理

通過上一個例子,我們還可以看出,為了使程序更健壯,我們還需要加入異常處理。 在異步的方式下,異常處理分布在不同的回調(diào)函數(shù)中,我們無法在調(diào)用的時候通過try…catch的方式來處理異常, 所以很難做到有效,清楚。

#p#

更好的Javascript異步編程方式

“這是***的時代,也是最糟糕的時代”

為了解決Javascript異步編程帶來的問題,很多的開發(fā)者做出了不同程度的努力,提供了很多不同的解決方案。然而面對如此眾多的方案應該如何選擇呢?我們這就來看看都有哪些可供選擇的方案吧。

Promise

Promise 對 象曾經(jīng)以多種形式存在于很多語言中。這個詞***由C++工程師用在Xanadu 項目中,Xanadu 項目是Web 應用項目的先驅(qū)。隨后Promise 被用在E編程語言中,這又激發(fā)了Python 開發(fā)人員的靈感,將它實現(xiàn)成了Twisted 框架的Deferred 對象。

2007 年,Promise 趕上了JavaScript 大潮,那時Dojo 框架剛從Twisted框架汲取靈感,新增了一個叫做dojo.Deferred 的對象。也就在那個時候,相對成熟的Dojo 框架與初出茅廬的jQuery 框架激烈地爭奪著人氣和名望。2009 年,Kris Zyp 有感于dojo.Deferred 的影響力提出了CommonJS 之Promises/A 規(guī)范。同年,Node.js ***亮相。

在 編程的概念中,future,promise,和delay表示同一個概念。Promise翻譯成中文是“承諾”,也就是說給你一個東西,我保證未來能夠 做到,但現(xiàn)在什么都沒有。它用來表示異步操作返回的一個對象,該對象是用來獲取未來的執(zhí)行結(jié)果的一個代理,初始值不確定。許多語言都有對Promise的 支持。

Promise的核心是它的then方法,我們可以使用這個方法從異步操作中得到返回值,或者是異常。then有兩個可選參數(shù)(有的實現(xiàn)是三個),分別處理成功和失敗的情景。

  1. var promise = doSomethingAync() 
  2. promise.then(onFulfilled, onRejected) 

異 步調(diào)用doSomethingAync返回一個Promise對象promise,調(diào)用promise的then方法來處理成功和失敗。這看上去似乎并沒 有很大的改進。仍然需要回調(diào)。但是和以前的區(qū)別在于,首先異步操作有了返回值,雖然該值只是一個對未來的承諾;其次通過使用then,程序員可以有效的控 制流程異常處理,決定如何使用這個來自未來的值。

對于嵌套的異步操作,有了Promise的支持,可以寫成這樣的鏈式操作:

  1. operation1().then(function (result1) { 
  2.     return operation2(result1) 
  3. }).then(function (result2) { 
  4.     return operation3(result2); 
  5. }).then(function (result3) { 
  6.     return operation4(result3); 
  7. }).then(function (result4) { 
  8.     return operation5(result4) 
  9. }).then(function (result5) { 
  10.     //And so on 
  11. }); 

Promise提供更便捷的流程控制,例如Promise.all()可以解決需要并發(fā)的執(zhí)行若干個異步操作,等所有操作完成后進行處理。

  1. var p1 = async1(); 
  2. var p2 = async2(); 
  3. var p3 = async3(); 
  4. Promise.all([p1,p2,p3]).then(function(){ 
  5.     // do something when all three asychronized operation finished 
  6. }); 

對于異常處理,

  1. doA() 
  2.   .then(doB) 
  3.   .then(null,function(error){ 
  4.       // error handling here 
  5.   }) 

如果doA失敗,它的Promise會被拒絕,處理鏈上的下一個onRejected會被調(diào)用,在這個例子中就是匿名函數(shù)function(error){}。比起原始的回調(diào)方式,不需要在每一步都對異常進行處理。這生了不少事。

以上只是對于Promise概念的簡單陳述,Promise擁有許多不同規(guī)范建議(A,A+,B,KISS,C,D等),名字(Future,Promise,Defer),和開源實現(xiàn)。大家可以參考一下的這些鏈接。

如果你有選擇困難綜合癥,面對這么多的開源庫不知道如何決斷,先不要急,這還只是一部分,還有一些庫沒有或者不完全采用Promise的概念

Non-Promise

下面列出了其它的一些開源的庫,也可以幫助解決Javascript中異步編程所遇到的諸多問題,它們的解決方案各不相同,我這里就不一一介紹了。大家有興趣可以去看看或者試用一下。

Non-3rd Party

其實,為了解決Javascript異步編程帶來的問題,不一定非要使用Promise或者其它的開源庫,這些庫提供了很好的模式,但是你也可以通過有針對性的設計來解決。

比如,對于層層回調(diào)的模式,可以利用消息機制來改寫,假定你的系統(tǒng)中已經(jīng)實現(xiàn)了消息機制,你的code可以寫成這樣:

  1. var co = require('co'); 
  2. var fs = require('fs'); 
  3.    
  4. var stat = function(path) { 
  5.   return function(cb){ 
  6.     fs.stat(path,cb); 
  7.   } 
  8. }; 
  9.    
  10. var readFile = function(filename) { 
  11.   return function(cb){ 
  12.     fs.readFile(filename,cb); 
  13.   } 
  14. }; 
  15.    
  16. co(function *() { 
  17.   var stat = yield stat('./README.md'); 
  18.   var content = yield readFile('./README.md'); 
  19. })(); 

這樣我們就把嵌套的異步調(diào)用,改寫成了順序執(zhí)行的事件處理。

更多的方式,請大家參考這篇文章,它提出了解決異步的五種模式:回調(diào)、觀察者模式(事件)、消息、Promise和有限狀態(tài)機(FSM)。

#p#

下一代Javscript對異步編程的增強

ECMAScript6

下一代的Javascript標準Harmony,也就是ECMAScript6正在醞釀中,它提出了許多新的語言特性,比如箭頭函數(shù)、類(Class)、生成器(Generator)、Promise等等。其中Generator和Promise都可以被用于對異步調(diào)用的增強。

Nodejs的開發(fā)版V0.11已經(jīng)可以支持ES6的一些新的特性,使用node –harmony命令來運行對ES6的支持。

co、Thunk、Koa

koa是由Express原班人馬(主要是TJ)打造,希望提供一個更精簡健壯的nodejs框架。koa依賴ES6中的Generator等新特性,所以必須運行在相應的Nodejs版本上。

利用Generator、coThunk,可以在Koa中有效的解決Javascript異步調(diào)用的各種問題。

co是一個異步流程簡化的工具,它利用Generator把一層層嵌套的調(diào)用變成同步的寫法。

  1. var co = require('co'); 
  2. var fs = require('fs'); 
  3.    
  4. var stat = function(path) { 
  5.   return function(cb){ 
  6.     fs.stat(path,cb); 
  7.   } 
  8. }; 
  9.    
  10. var readFile = function(filename) { 
  11.   return function(cb){ 
  12.     fs.readFile(filename,cb); 
  13.   } 
  14. }; 
  15.    
  16. co(function *() { 
  17.   var stat = yield stat('./README.md'); 
  18.   var content = yield readFile('./README.md'); 
  19. })(); 

通過co可以把異步的fs.readFile當成同步一樣調(diào)用,只需要把異步函數(shù)fs.readFile用閉包的方式封裝。

利用Thunk可以進一步簡化為如下的code, 這里Thunk的作用就是用閉包封裝異步函數(shù),返回一個生成函數(shù)的函數(shù),供生成器來調(diào)用。

利用co可以串行或者并行的執(zhí)行異步調(diào)用。

  1. var thunkify = require('thunkify'); 
  2. var co = require('co'); 
  3. var fs = require('fs'); 
  4.    
  5. var stat = thunkify(fs.stat); 
  6. var readFile = thunkify(fs.readFile); 
  7.    
  8. co(function *() { 
  9.   var stat = yield stat('./README.md'); 
  10.   var content = yield readFile('./README.md'); 
  11. })(); 

串行

  1. co(function *() { 
  2.   var a = yield request(a); 
  3.   var b = yield request(b); 
  4. })(); 

并行

  1. co(function *() { 
  2.  var res = yield [request(a), request(b)]; 
  3. })(); 

更多詳細的內(nèi)容,大家可以參考這兩篇文章12

總結(jié)

異 步編程帶來的問題在客戶端Javascript中并不明顯,但隨著服務器端Javascript越來越廣的被使用,大量的異步IO操作使得該問題變得明 顯。許多不同的方法都可以解決這個問題,本文討論了一些方法,但并不深入。大家需要根據(jù)自己的情況選擇一個適于自己的方法。

同時,隨著ES6的定義,Javascript的語法變得越來越豐富,更多的功能帶來了很多便利,然而原本簡潔,單一目的的Javascript變得復雜,也要承擔更多的任務。Javascript何去何從,讓我們拭目以待。

 

責任編輯:王雪燕 來源: 博客園
相關推薦

2014-05-23 10:12:20

Javascript異步編程

2020-10-15 13:29:57

javascript

2017-07-13 12:12:19

前端JavaScript異步編程

2016-09-07 20:43:36

Javascript異步編程

2011-11-11 15:47:22

JavaScript

2021-06-02 09:01:19

JavaScript 前端異步編程

2021-10-15 09:56:10

JavaScript異步編程

2021-12-10 07:47:30

Javascript異步編程

2011-07-27 14:10:43

javascript

2022-10-31 09:00:24

Promise數(shù)組參數(shù)

2011-11-10 10:23:56

Jscex

2023-12-04 13:22:00

JavaScript異步編程

2021-06-06 19:51:07

JavaScript異步編程

2013-04-01 15:38:54

異步編程異步編程模型

2013-03-08 09:33:25

JavaScript同步異步

2016-10-21 11:04:07

JavaScript異步編程原理解析

2013-01-07 10:44:00

JavaScriptjQueryJS

2021-11-01 22:36:04

JavaScript

2011-11-17 16:14:25

Jscex

2011-11-11 13:38:39

Jscex
點贊
收藏

51CTO技術(shù)棧公眾號

国产精品一区二区中文字幕 | av综合网站| 夜夜爽夜夜爽精品视频| 国产成人综合亚洲| 2020中文字字幕在线不卡| 日韩成人动漫在线观看| 一本大道久久精品懂色aⅴ| 国产在线观看一区二区三区| 在线免费观看h| 99精品桃花视频在线观看| 国产精品视频500部| 三级国产三级在线| 91在线免费播放| 日韩av免费电影| 精品久久视频| 日韩一中文字幕| 欧美婷婷久久五月精品三区| 2020国产精品| 97蝌蚪自拍自窝| 亚洲激情成人| 性色av一区二区三区免费| 在线天堂资源| 久久精品视频中文字幕| 日韩一区二区三区在线免费观看 | 日韩有码欧美| 欧美一级一级性生活免费录像| h无码动漫在线观看| 成人在线免费小视频| 久久精品国产精品亚洲| 国产国产国产国产国产国产| 久久xxxx精品视频| 中文视频一区视频二区视频三区| 蜜桃av综合| 亚洲国产欧美不卡在线观看 | 中文字幕日韩高清| 在线欧美一区二区| 国产女人18毛片| 激情综合色播激情啊| 亚洲电影网站| 中文在线一区二区| 欧美色图色综合| 自拍偷在线精品自拍偷无码专区| 青青草视频在线观看| 蜜桃av一区二区在线观看| 久久综合伊人77777蜜臀| 手机在线看福利| 蜜臂av日日欢夜夜爽一区| 成人伊人精品色xxxx视频| 波多野结衣在线观看| 欧美日韩在线第一页| 99色这里只有精品| 亚洲深夜福利| 国产精品久久国产精品| 欧美a在线观看| 中文字幕亚洲国产| 岛国av在线网站| 欧美一区永久视频免费观看| 你懂的视频在线观看| 国产精品乱人伦| 亚洲欧美在线网| 日韩国产在线一| www.日本三级| 亚洲图片欧美一区| 69视频在线| 日韩精品在线免费观看| 亚洲三级一区| www.av91| 久久久精品2019中文字幕之3| 久久久999精品免费| 国产日产欧美一区| 欧美三级电影在线播放| 黄色亚洲在线| 国内精品**久久毛片app| 91综合网人人| 日本一区二区久久精品| 欧美亚洲成人xxx| 日本一区二区三区视频在线| 亚洲国产精品大全| 偷拍视频一区二区三区| 久久久久久成人精品| 国产精品18hdxxxⅹ在线| 海角国产乱辈乱精品视频| 欧美三级三级| 你懂的视频在线一区二区| 亚洲精品国偷自产在线99热| 亚洲欧美另类久久久精品2019| 国产精品欧美激情在线观看 | 老**午夜毛片一区二区三区| 久久精品国产99精品国产亚洲性色| 欧美三级美国一级| 日本福利一区二区三区| 国产精品99久久久久久有的能看 | 亚洲www.| 97av在线视频| 三级欧美在线一区| 欧美日韩国产中文字幕在线| 色欧美日韩亚洲| 国产黄色特级片| 一本岛在线视频| 久久综合资源网| 超碰在线观看免费版| 91精品中国老女人| 亚洲精品日本| 偷窥自拍亚洲色图| 欧美一级在线观看| 久久精品福利| 精品成人一区二区三区免费视频| 成人h猎奇视频网站| 国产成人免费视频| h视频在线免费观看| 国产精品成人观看视频国产奇米| 欧美在线高清视频| av午夜在线| 欧美精品一区在线播放| 99成人在线视频| 国产亚洲综合视频| 欧美一卡2卡三卡4卡5免费| 日本电影一区二区| 国产精品一线二线三线| 欧美亚洲一区二区三区四区| av综合网址| 手机看片一级片| 日韩中文字幕亚洲| 国产剧情av麻豆香蕉精品| 影音先锋在线播放| 蜜桃成人在线| 国产精品678| 久久久久久免费毛片精品| 亚洲精品一二三**| 亚洲综合色在线观看| 亚洲福利视频久久| 国内精品视频一区二区三区八戒| 日韩毛片久久久| 亚洲综合久久av| 26uuu国产精品视频| 成人国产精品视频| 韩国女主播一区二区三区| 国产日韩一区二区在线观看| 一区二区三区在线视频观看58| h视频在线播放| 中文字幕剧情在线观看一区| 91精品国产综合久久久久久 | 一本色道久久综合狠狠躁的推荐| 日本亚洲欧洲无免费码在线| 台湾色综合娱乐中文网| 亚洲综合精品自拍| 亚洲欧美视频在线观看| 成人综合网址| 国产欧美在线看| 色黄久久久久久| 国产a一区二区| 亚洲一区中文字幕| 亚洲无线码在线一区观看| 亚洲欧美日韩精品| 大陆精大陆国产国语精品| 天堂资源在线观看| 91精品国产高清久久久久久91| 国产精品爱久久久久久久| 国产精品美女黄网| 国产亚洲精品网站| 99re在线视频免费观看| 91精品国产一区二区三区动漫| 久久久精品影院| 91视频精品在这里| 麻豆久久久久久久| 久久爱另类一区二区小说| 久久精品免费看| 国产精品国产三级国产a| 在线观看视频一区二区欧美日韩| 精品国产污污免费网站入口 | 色综合视频一区中文字幕| 91精品欧美综合在线观看最新| 国产一区二区三区免费| 欧美伊人影院| 亚洲另类一区二区| 国产精品久久久久久亚洲伦| 亚洲午夜激情av| 亚洲女人被黑人巨大进入| 欧美麻豆久久久久久中文| 欧美日韩不卡在线| 国产午夜精品久久久 | 国产乱视频在线观看| 国产精品视频yy9099| 国产精品视频久久| wwww.国产| 调教一区二区| 精品国产乱码久久久久久蜜坠欲下| 欧美日韩三级电影在线| 尹人成人综合网| 日本一二三四高清不卡| 久久久精品国产一区二区| 欧美做暖暖视频| 成人影院网站| 国产精品v亚洲精品v日韩精品| 中文字幕精品—区二区四季| 欧美日韩国产成人在线| 性高湖久久久久久久久aaaaa| 成人在线免费观看av| 日本亚洲欧洲无免费码在线| 懂色av一区二区三区蜜臀|