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

基于事件的JavaScript編程:異步與同步

開發(fā) 前端
JavaScript最基礎的異步函數(shù)是setTimeout和setInterval。setTimeout會在一定時間后執(zhí)行給定的函數(shù)。它接受一個回調函數(shù)作為第一參數(shù)和一個毫秒時間作為第二參數(shù)。

JavaScript的優(yōu)勢之一是其如何處理異步代碼。異步代碼會被放入一個事件隊列,等到所有其他代碼執(zhí)行后才進行,而不會阻塞線程。然而,對于初學者來說,書寫異步代碼可能會比較困難。而在這篇文章里,我將會消除你可能會有的任何困惑。

理解異步代碼

JavaScript最基礎的異步函數(shù)是setTimeoutsetInterval。setTimeout會在一定時間后執(zhí)行給定的函數(shù)。它接受一個回調函數(shù)作為***參數(shù)和一個毫秒時間作為第二參數(shù)。以下是用法舉例:

  1. console.log( "a" );  
  2. setTimeout(function() {  
  3.     console.log( "c" )  
  4. }, 500 );  
  5. setTimeout(function() {  
  6.     console.log( "d" )  
  7. }, 500 );  
  8. setTimeout(function() {  
  9.     console.log( "e" )  
  10. }, 500 );  
  11. console.log( "b" ); 

正如預期,控制臺先輸出“a”、“b”,大約500毫秒后,再看到“c”、“d”、“e”。我用“大約”是因為setTimeout事實上是不可預知的。實際上,甚至 HTML5規(guī)范都提到了這個問題:

“這個API不能保證計時會如期準確地運行。由于CPU負載、其他任務等所導致的延遲是可以預料到的。”

有趣的是,直到在同一程序段中所有其余的代碼執(zhí)行結束后,超時才會發(fā)生。所以如果設置了超時,同時執(zhí)行了需長時間運行的函數(shù),那么在該函數(shù)執(zhí)行完成之前,超時甚至都不會啟動。實際上,異步函數(shù),如setTimeout和setInterval,被壓入了稱之為Event Loop的隊列。

Event Loop是一個回調函數(shù)隊列。當異步函數(shù)執(zhí)行時,回調函數(shù)會被壓入這個隊列。JavaScript引擎直到異步函數(shù)執(zhí)行完成后,才會開始處理事件循環(huán)。這意味著JavaScript代碼不是多線程的,即使表現(xiàn)的行為相似。事件循環(huán)是一個先進先出(FIFO)隊列,這說明回調是按照它們被加入隊列的順序執(zhí)行的。JavaScript被 node選做為開發(fā)語言,就是因為寫這樣的代碼多么簡單啊。

Ajax

異步Javascript與XML(AJAX)***性的改變了Javascript語言的狀況。突然間,瀏覽器不再需要重新加載即可更新web頁面。 在不同的瀏覽器中實現(xiàn)Ajax的代碼可能漫長并且乏味;但是,幸虧有jQuery(還有其他庫)的幫助,我們能夠以很容易并且優(yōu)雅的方式實現(xiàn)客戶端-服務器端通訊。

我們可以使用jQuery跨瀏覽器接口$.ajax很容易地檢索數(shù)據,然而卻不能呈現(xiàn)幕后發(fā)生了什么。比如:

  1. var data;  
  2. $.ajax({  
  3.     url: "some/url/1",  
  4.     success: function( data ) {  
  5.         // But, this will!  
  6.         console.log( data );  
  7.     }  
  8. })  
  9. // Oops, this won't work...  
  10. console.log( data ); 

較容易犯的錯誤,是在調用$.ajax之后馬上使用data,但是實際上是這樣的:

  1. xmlhttp.open( "GET""some/ur/1"true );  
  2. xmlhttp.onreadystatechange = function( data ) {  
  3.     if ( xmlhttp.readyState === 4 ) {  
  4.         console.log( data );  
  5.     }  
  6. };  
  7. xmlhttp.send( null ); 

底層的XmlHttpRequest對象發(fā)起請求,設置回調函數(shù)用來處理XHR的readystatechnage事件。然后執(zhí)行XHR的send方法。在XHR運行中,當其屬性readyState改變時readystatechange事件就會被觸發(fā),只有在XHR從遠端服務器接收響應結束時回調函數(shù)才會觸發(fā)執(zhí)行。

處理異步代碼

異步編程很容易陷入我們常說的“回調地獄”。因為事實上幾乎JS中的所有異步函數(shù)都用到了回調,連續(xù)執(zhí)行幾個異步函數(shù)的結果就是層層嵌套的回調函數(shù)以及隨之而來的復雜代碼。

node.js中的許多函數(shù)也是異步的。因此如下的代碼基本上很常見:

  1. var fs = require( "fs" );  
  2. fs.exists( "index.js"function() {  
  3.     fs.readFile( "index.js""utf8"function( err, contents ) {  
  4.         contents = someFunction( contents ); // do something with contents  
  5.         fs.writeFile( "index.js""utf8"function() {  
  6.             console.log( "whew! Done finally..." );  
  7.         });  
  8.     });  
  9. });  
  10. console.log( "executing..." ); 

下面的客戶端代碼也很多見:

  1. GMaps.geocode({  
  2.     address: fromAddress,  
  3.     callback: function( results, status ) {  
  4.         if ( status == "OK" ) {  
  5.             fromLatLng = results[0].geometry.location;  
  6.             GMaps.geocode({  
  7.                 address: toAddress,  
  8.                 callback: function( results, status ) {  
  9.                     if ( status == "OK" ) {  
  10.                         toLatLng = results[0].geometry.location;  
  11.                         map.getRoutes({  
  12.                             origin: [ fromLatLng.lat(), fromLatLng.lng() ],  
  13.                             destination: [ toLatLng.lat(), toLatLng.lng() ],  
  14.                             travelMode: "driving",  
  15.                             unitSystem: "imperial",  
  16.                             callback: function( e ){  
  17.                                 console.log( "ANNNND FINALLY here's the directions..." );  
  18.                                 // do something with e  
  19.                             }  
  20.                         });  
  21.                     }  
  22.                 }  
  23.             });  
  24.         }  
  25.     }  
  26. }); 

Nested callbacks can get really nasty, but there are several solutions to this style of coding.

嵌套的回調很容易帶來代碼中的“壞味道”,不過你可以用以下的幾種風格來嘗試解決這個問題

The problem isn’t with the language itself; it’s with the way programmers use the language — Async Javascript.

沒有糟糕的語言,只有糟糕的程序猿 ——異步JavaSript

命名函數(shù)

清除嵌套回調的一個便捷的解決方案是簡單的避免雙層以上的嵌套。傳遞一個命名函數(shù)給作為回調參數(shù),而不是傳遞匿名函數(shù):

  1. var fromLatLng, toLatLng;  
  2. var routeDone = function( e ){  
  3.     console.log( "ANNNND FINALLY here's the directions..." );  
  4.     // do something with e  
  5. };  
  6. var toAddressDone = function( results, status ) {  
  7.     if ( status == "OK" ) {  
  8.         toLatLng = results[0].geometry.location;  
  9.         map.getRoutes({  
  10.             origin: [ fromLatLng.lat(), fromLatLng.lng() ],  
  11.             destination: [ toLatLng.lat(), toLatLng.lng() ],  
  12.             travelMode: "driving",  
  13.             unitSystem: "imperial",  
  14.             callback: routeDone  
  15.         });  
  16.     }  
  17. };  
  18. var fromAddressDone = function( results, status ) {  
  19.     if ( status == "OK" ) {  
  20.         fromLatLng = results[0].geometry.location;  
  21.         GMaps.geocode({  
  22.             address: toAddress,  
  23.             callback: toAddressDone  
  24.         });  
  25.     }  
  26. };  
  27. GMaps.geocode({  
  28.     address: fromAddress,  
  29.     callback: fromAddressDone  
  30. }); 

此外, async.js 庫可以幫助我們處理多重Ajax requests/responses. 例如:

  1. async.parallel([  
  2.     function( done ) {  
  3.         GMaps.geocode({  
  4.             address: toAddress,  
  5.             callback: function( result ) {  
  6.                 done( null, result );  
  7.             }  
  8.         });  
  9.     },  
  10.     function( done ) {  
  11.         GMaps.geocode({  
  12.             address: fromAddress,  
  13.             callback: function( result ) {  
  14.                 done( null, result );  
  15.             }  
  16.         });  
  17.     }  
  18. ], function( errors, results ) {  
  19.     getRoute( results[0], results[1] );  
  20. }); 

這段代碼執(zhí)行兩個異步函數(shù),每個函數(shù)都接收一個名為"done"的回調函數(shù)并在函數(shù)結束的時候調用它。當兩個"done"回調函數(shù)結束后,parallel函數(shù)的回調函數(shù)被調用并執(zhí)行或處理這兩個異步函數(shù)產生的結果或錯誤。

Promises模型

引自 CommonJS/A

promise表示一個操作獨立完成后返回的最終結果。

有很多庫都包含了promise模型,其中jQuery已經有了一個可使用且很出色的promise API。jQuery在1.5版本引入了Deferred對象,并可以在返回promise的函數(shù)中使用jQuery.Deferred的構造結果。而返回promise的函數(shù)則用于執(zhí)行某種異步操作并解決完成后的延遲。

  1. var geocode = function( address ) {  
  2.     var dfd = new $.Deferred();  
  3.     GMaps.geocode({  
  4.         address: address,  
  5.         callback: function( response, status ) {  
  6.             return dfd.resolve( response );  
  7.         }  
  8.     });  
  9.     return dfd.promise();  
  10. };  
  11. var getRoute = function( fromLatLng, toLatLng ) {  
  12.     var dfd = new $.Deferred();  
  13.     map.getRoutes({  
  14.         origin: [ fromLatLng.lat(), fromLatLng.lng() ],  
  15.         destination: [ toLatLng.lat(), toLatLng.lng() ],  
  16.         travelMode: "driving",  
  17.         unitSystem: "imperial",  
  18.         callback: function( e ) {  
  19.             return dfd.resolve( e );  
  20.         }  
  21.     });  
  22.     return dfd.promise();  
  23. };  
  24. var doSomethingCoolWithDirections = function( route ) {  
  25.     // do something with route  
  26. };  
  27. $.when( geocode( fromAddress ), geocode( toAddress ) ).  
  28.     then(function( fromLatLng, toLatLng ) {  
  29.         getRoute( fromLatLng, toLatLng ).then( doSomethingCoolWithDirections );  
  30.     }); 

這允許你執(zhí)行兩個異步函數(shù)后,等待它們的結果,之后再用先前兩個調用的結果來執(zhí)行另外一個函數(shù)。

promise表示一個操作獨立完成后返回的最終結果。

在這段代碼里,geocode方法執(zhí)行了兩次并返回了一個promise。異步函數(shù)之后執(zhí)行,并在其回調里調用了resolve。然后,一旦兩次調用resolve完成,then將會執(zhí)行,其接收了之前兩次調用geocode的返回結果。結果之后被傳入getRoute,此方法也返回一個promise。最終,當getRoute的promise解決后,doSomethingCoolWithDirections回調就執(zhí)行了。

事件

事件是另一種當異步回調完成處理后的通訊方式。一個對象可以成為發(fā)射器并派發(fā)事件,而另外的對象則監(jiān)聽這些事件。這種類型的事件處理方式稱之為 觀察者模式backbone.js 庫在withBackbone.Events中就創(chuàng)建了這樣的功能模塊。

  1. var SomeModel = Backbone.Model.extend({  
  2.    url: "/someurl" 
  3. });  
  4. var SomeView = Backbone.View.extend({  
  5.     initialize: function() {  
  6.         this.model.on( "reset"this.render, this );  
  7.         this.model.fetch();  
  8.     },  
  9.     render: function( data ) {  
  10.         // do something with data  
  11.     }  
  12. });  
  13. var view = new SomeView({  
  14.     model: new SomeModel()  
  15. }); 

還有其他用于發(fā)射事件的混合例子和函數(shù)庫,例如 jQuery Event Emitter EventEmitter monologue.js ,以及node.js內建的 EventEmitter 模塊。

事件循環(huán)是一個回調函數(shù)的隊列。

一個類似的派發(fā)消息的方式稱為 中介者模式 postal.js 庫中用的即是這種方式。在中介者模式,有一個用于所有對象監(jiān)聽和派發(fā)事件的中間人。在這種模式下,一個對象不與另外的對象產生直接聯(lián)系,從而使得對象間都互相分離。

絕不要返回promise到一個公用的API。這不僅關系到了API用戶對promises的使用,也使得重構更加困難。不過,內部用途的promises和外部接口的事件的結合,卻可以讓應用更低耦合且便于測試。

在先前的例子里面,doSomethingCoolWithDirections回調函數(shù)在兩個geocode函數(shù)完成后執(zhí)行。然后,doSomethingCoolWithDirections才會獲得從getRoute接收到的響應,再將其作為消息發(fā)送出去。

  1. var doSomethingCoolWithDirections = function( route ) {  
  2.     postal.channel( "ui" ).publish( "directions.done",  {  
  3.         route: route  
  4.     });  
  5. }; 

這允許了應用的其他部分不需要直接引用產生請求的對象,就可以響應異步回調。而在取得命令時,很可能頁面的好多區(qū)域都需要更新。在一個典型的jQuery Ajax過程中,當接收到的命令變化時,要順利的回調可能就得做相應的調整了。這可能會使得代碼難以維護,但通過使用消息,處理UI多個區(qū)域的更新就會簡單得多了。

  1. var UI = function() {  
  2.     this.channel = postal.channel( "ui" );  
  3.     this.channel.subscribe( "directions.done"this.updateDirections ).withContext( this );  
  4. };  
  5. UI.prototype.updateDirections = function( data ) {  
  6.     // The route is available on data.route, now just update the UI  
  7. };  
  8. app.ui = new UI(); 

另外一些基于中介者模式傳送消息的庫有 amplify, PubSubJS, and radio.js。 

結論

JavaScript 使得編寫異步代碼很容易. 使用 promises, 事件, 或者命名函數(shù)來避免“callback hell”. 為獲取更多javascript異步編程信息,請點擊Async JavaScript: Build More Responsive Apps with Less . 更多的實例托管在github上,地址NetTutsAsyncJS,趕快Clone吧 !

原文鏈接:http://www.oschina.net/translate/event-based-programming-what-async-has-over-sync

責任編輯:張偉 來源: oschina
相關推薦

2013-04-01 15:25:41

異步編程異步EMP

2017-07-13 12:12:19

前端JavaScript異步編程

2009-08-20 17:47:54

C#異步編程模式

2020-10-15 13:29:57

javascript

2010-04-06 15:20:56

ASP.NET MVC

2015-09-07 14:08:32

Java編程異步事件

2016-09-07 20:43:36

Javascript異步編程

2015-04-22 10:50:18

JavascriptJavascript異

2014-05-23 10:12:20

Javascript異步編程

2021-10-22 08:29:14

JavaScript事件循環(huán)

2021-12-10 07:47:30

Javascript異步編程

2021-10-15 09:56:10

JavaScript異步編程

2011-11-11 15:47:22

JavaScript

2021-06-02 09:01:19

JavaScript 前端異步編程

2023-09-06 09:00:00

架構開發(fā)異步編程

2013-01-07 10:44:00

JavaScriptjQueryJS

2014-12-17 09:58:16

2012-07-27 10:02:39

C#

2012-03-01 20:32:29

iOS

2017-05-11 20:20:59

JavascriptPromiseWeb
點贊
收藏

51CTO技術棧公眾號

777视频在线观看| 国产激情小视频在线| 日韩中文字幕亚洲一区二区va在线| 中文字幕精品在线| 污网站免费在线观看| 国产精品免费视频网站| 久久av综合网| 久久99精品久久久久久久久久久久| 超碰97国产在线| 99视频精品全国免费| 国产精品你懂得| 日韩电影不卡一区| 91精品国产高清| 九九热播视频在线精品6| 欧美大尺度在线观看| 成人不卡视频| 国产亚洲人成网站在线观看 | 国产精品美女久久久| 91人人爽人人爽人人精88v| 一区二区三区视频免费观看| 高清欧美性猛交xxxx| 粉嫩的18在线观看极品精品| 欧美日本亚洲视频| 试看120秒一区二区三区| 欧美成人在线免费视频| 狂野欧美xxxx韩国少妇| 久久久久亚洲精品| 伊甸园亚洲一区| 91中文字幕一区| 亚洲中午字幕| 亚洲成人午夜在线| 国产一区二区在线看| 日本福利视频在线| 欧美国产日本韩| 嫩草影院2018| 香蕉影视欧美成人| 拍真实国产伦偷精品| 亚洲第一区第二区| 韩日成人影院| 欧美大奶子在线| 精品国产一区二区三区| 97人人模人人爽人人喊38tv| 国产精品视区| 午夜久久久久久久久久久| 99re这里只有精品视频首页| 97视频网站| 日本福利一区二区| 日韩经典av| 久久久999精品视频| 亚洲毛片免费看| 国产伦精品一区二区三区视频免费 | 亚洲成人看片| 欧美精品在线播放| 久久不见久久见免费视频7| 91丝袜美腿美女视频网站| 国产日韩欧美一区在线| 四虎影院一区二区| 中文字幕亚洲区| 尤物视频在线免费观看| 国产一区二区久久精品| 妖精一区二区三区精品视频 | 女一区二区三区| 国产精品国产三级欧美二区| 久久99热这里只有精品| 国产美女三级视频| 色综合天天性综合| 神马久久资源| 国产精品久久久久久搜索 | wwwwww.欧美系列| 亚洲欧美激情小说另类| 亚洲热app| 亚洲欧洲一区二区三区久久| 宅男在线一区| 亚洲日本精品| 成人免费小视频| 国产探花在线观看| 日本不卡高字幕在线2019| 亚洲成人在线| 青青在线视频免费| 91精品国产色综合久久不卡电影| free欧美| ts人妖另类在线| 国产午夜精品久久久久久久| 国产精品久久久久久福利| 久久久伊人欧美| 青青草97国产精品免费观看无弹窗版| 99热最新网址| 亚洲欧美精品suv| 久久美女精品| 欧美 日韩精品| 日韩欧美国产精品| 成人激情开心网| 成熟了的熟妇毛茸茸| 3d成人动漫网站| 国产99久久久国产精品成人免费| 五月天综合婷婷| 欧美亚洲综合色| 精品大片一区二区| 九色porny91| 在线成人免费网站| 亚洲一区激情| 在线视频99| 久久777国产线看观看精品| 免费在线看成人av| 日韩一二三四| 日本成人精品在线| 91蜜桃在线观看| 欧美大胆a人体大胆做受| 91亚洲精品在线| 国产精品亲子伦对白| 国产精品.xx视频.xxtv| 亚洲v国产v在线观看| 欧美私人免费视频| 国产精品7m凸凹视频分类| 1pon在线| 97精品视频在线观看| 久久这里都是精品| 成人在线免费| h无码动漫在线观看| 日韩成人免费视频| 日本va欧美va欧美va精品| 毛片av在线| 99国精产品一二二线| 超碰在线免费播放| 清纯唯美日韩制服另类| 无遮挡亚洲一区| avtt亚洲| 国产91精品在线播放| 99久久综合国产精品| 欧美高清另类hdvideosexjaⅴ| 99re6热在线精品视频播放速度| 亚洲素人一区二区| 色狼人综合干| 国产美女免费观看| 国产精品精品国产| 午夜精品一区在线观看| 欧美精品乱码| 一级片在线视频| 亚洲一区精品电影| 91电影在线观看| 伊人久久大香线| 色欧美激情视频在线| 国内精品视频免费| 91精品国产日韩91久久久久久| 久久国产精品久久久久久电车 | 久久字幕精品一区| 成人av黄色| 一区二区三区视频在线播放| 国产婷婷色综合av蜜臀av | 久久av一区二区三区亚洲| 欧美日韩亚洲综合在线 欧美亚洲特黄一级| 午夜片欧美伦| 国产人成在线观看| 国内精品久久久久久久果冻传媒| 欧美乱妇23p| 国产一本一道久久香蕉| av国产精品| 国产国语**毛片高清视频| 国产综合视频在线观看| 91成人看片片| 蜜桃精品视频在线| 国产精品久久久久久久久免费高清 | 欧美xxxx免费虐| 午夜精品亚洲一区二区三区嫩草| 亚洲国产古装精品网站| 91啪九色porn原创视频在线观看| 91精品国产自产精品男人的天堂| av天天在线| 精品亚洲欧美日韩| 亚洲精品怡红院| 欧美激情a在线| 午夜欧美2019年伦理| 首页国产欧美日韩丝袜| 成人天堂yy6080亚洲高清| 91av在线免费播放| 国产综合香蕉五月婷在线| 精品99999| 中文字幕av不卡| 欧美三区美女| 午夜精品成人av| 最新中文字幕av专区| 午夜老司机精品| 奇米亚洲欧美| 蜜桃视频在线观看www社区| 国产在线无码精品| 国产精品黄色影片导航在线观看| 欧美mv日韩mv国产网站app| 欧美国产精品一区二区| 一本色道久久综合亚洲精品高清 | 亚洲午夜免费电影| 免费视频最近日韩| 国内精品国产成人国产三级粉色| 成人免费在线观看| 成人毛片视频网站| 99porn视频在线| 日日狠狠久久偷偷四色综合免费 | 免费高清视频日韩| 亚洲国产精品久久久久久女王| 国产z一区二区三区| 精品视频在线导航|