成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

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

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

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

理解異步代碼

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

  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規范都提到了這個問題:

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

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

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

Ajax

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

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

  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對象發起請求,設置回調函數用來處理XHR的readystatechnage事件。然后執行XHR的send方法。在XHR運行中,當其屬性readyState改變時readystatechange事件就會被觸發,只有在XHR從遠端服務器接收響應結束時回調函數才會觸發執行。

處理異步代碼

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

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

  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

命名函數

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

  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. }); 

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

Promises模型

引自 CommonJS/A

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

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

  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.     }); 

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

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

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

事件

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

  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. }); 

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

事件循環是一個回調函數的隊列。

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

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

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

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

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

  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, 事件, 或者命名函數來避免“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

2015-09-07 14:08:32

Java編程異步事件

2010-04-06 15:20:56

ASP.NET MVC

2015-04-22 10:50:18

JavascriptJavascript異

2014-05-23 10:12:20

Javascript異步編程

2016-09-07 20:43:36

Javascript異步編程

2021-10-22 08:29:14

JavaScript事件循環

2021-12-10 07:47:30

Javascript異步編程

2021-06-02 09:01:19

JavaScript 前端異步編程

2021-10-15 09:56:10

JavaScript異步編程

2011-11-11 15:47:22

JavaScript

2023-09-06 09:00:00

架構開發異步編程

2013-01-07 10:44:00

JavaScriptjQueryJS

2014-12-17 09:58:16

2012-03-01 20:32:29

iOS

2012-07-27 10:02:39

C#

2017-05-11 20:20:59

JavascriptPromiseWeb
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 欧美精品影院 | 中文字幕欧美在线观看 | 韩日一区 | 四虎午夜剧场 | 亚洲国产免费 | 午夜精品久久久久久久久久久久久 | 国产亚洲一区二区精品 | 久久久中文 | 久久夜视频 | 欧美美女二区 | 欧美精品第一页 | 天天综合永久 | 国产精品久久久久久久久久久久久 | 日韩欧美在线一区 | www.日本在线播放 | 成人h动漫精品一区二区器材 | av在线免费观看网址 | 国产福利视频 | 久久精品一二三影院 | 高清免费在线 | 精品伊人 | 久久人体视频 | 国外成人在线视频 | 欧美激情视频一区二区三区免费 | 中文字幕二区三区 | 一级在线免费观看 | 久草福利| 国产91在线观看 | 精产国产伦理一二三区 | 欧美日韩黄色一级片 | 91在线看视频 | 91av免费观看| 欧美一区二区三区 | 黄网站涩免费蜜桃网站 | 日韩中文字幕在线观看 | 亚洲国产精品久久 | 久久不卡区 | 国产在线a视频 | 国产成人精品a视频一区www | 特级毛片爽www免费版 | 亚洲xx在线 |