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

F#與ASP.NET:基于事件的異步模式與異步Action

開發 開發工具 后端
提高ASP.NET應用程序伸縮性的有效手段之一便是使用異步請求,而在ASP.NET MVC 1中是不能直接支持異步Action的,因此我們需要使用一些簡單的Hack方式來實現這一點。

盡管在ASP.NET MVC 1中是不能直接支持異步Action,但在ASP.NET MVC 2中已經正式支持ASP.NET中的異步請求處理方式,并且通過一種比較易于使用的方式提供給開發人員使用。只可惜,由于語言層面的約束,這種使用方式還是有些不便,而此時便是F#的用武之地了。

基于事件的異步模式

說起.NET中的異步編程模型,.NET程序員最熟悉的應該就是Begin/End方法了。例如在WebRequest類中,便有這樣一對方法:

  1. var request = WebRequest.Create("http://www.ekrvqnd.cn/");  
  2. request.BeginGetResponse(ar => 
  3. {  
  4.     var response = request.EndGetResponse(ar);  
  5.     // use the response object  
  6.  
  7. }, null);  

在調用WebRequest對象的BeginGetResponse方法之后,當前調用線程不會被阻塞,而在異步操作完成之后,便會調用一個回調函數(即這里使用Lambda表達式構造的代碼快)進行通知,在這個回調函數中調用EndGetResponse方法便可以得到一個WebResponse對象作為結果。

在這個異步操作中,由于偉大的IOCP,我們可以使用極少數的線程同時發起成千上萬個連接(豪不夸張,我曾經在IIS里進行Comet試驗,同時建立起超過2w個連接進行通信)。不過,事實上在.NET中還有一種基于事件的異步模式(Event-based Asynchronous Pattern,EAP)。基于事件的異步編程的典型案例之一便是WebClient類:

  1. var client = new WebClient();  
  2. client.DownloadStringCompleted += (sender, args) => 
  3. {  
  4.     var html = args.Result;  
  5.     // ...  
  6. };  
  7.  
  8. client.DownloadStringAsync(new Uri("http://www.ekrvqnd.cn/"));  

基于事件的異步模式的關鍵便在于,它是使用事件來作為工作結束時的通知機制。它和Begin/End的異步模型有明顯區別。例如,在發生錯誤時,對于Begin/End模型來說會在End方法調用時拋出異常,而對于基于事件的異步模式來說,它則是使用事件參數的Exception屬性來告訴程序員是否有異常發生。如果Exception屬性為null,則說明一切正常,否則它便返回異步調用過程中發生的異常。

在ASP.NET MVC中使用異步Action

當年我的Hack使用的是Begin/End異步編程模型,而ASP.NET MVC 2則使用了基于事件的異步模式。圍繞這種模式,ASP.NET MVC的AsyncController還提供了相關的輔助方法,讓異步Action的編寫變得相對容易一些。這里我則直接引用MSDN上的示例來說明問題。首先,我們準備一個普通的同步Action:

 

  1. public class PortalController : Controller  
  2. {  
  3.     public ActionResult News(string city)  
  4.     {  
  5.         var newnewsService = new NewsService();  
  6.         var headlines = newsService.GetHeadlines(city);  
  7.         return View(headlines);  
  8.     }  

與它等價的異步Action則為:

  1. public class PortalController : AsyncController  
  2. {  
  3.     public void NewsAsync(string city)  
  4.     {  
  5.         AsyncManager.OutstandingOperations.Increment();  
  6.  
  7.         var newnewsService = new NewsService();  
  8.         newsService.GetHeadlinesCompleted += (sender, e) => 
  9.         {  
  10.             AsyncManager.Parameters["headlines"] = e.Value;  
  11.             AsyncManager.OutstandingOperations.Decrement();  
  12.         };  
  13.  
  14.         newsService.GetHeadlinesAsync(city);  
  15.     }  
  16.  
  17.     public ActionResult NewsCompleted(string[] headlines)  
  18.     {  
  19.         return View("News", headlines);  
  20.     }  
  21. }  

很顯然,異步Action也是標準的二段式調用,不過這個二段式調用卻由比較特別的“約定”。在ASP.NET MVC 2中使用異步Action時,首先需要繼承AsyncController類,并構造XyzAsync及XyzCompleted兩個方法,前者返回void,后者返回ActionResult——這便表示一個異步的Action,名為Xyz。

ASP.NET MVC 2中對于異步Action的開發也提供了一定支持,這個支持便來自于AsyncManager。在發起異步操作之前,我們可以調用其OutstandingOperations對象的Increment方法,表示需要“進行幾次異步操作”。

而每次異步操作結束之后,也就是在事件的處理函數中,便會調用對應的Decrement方法。這個方法表示“完成了一次異步操作”,而Decrement至零之后ASP.NET MVC便會得知所有的異步操作已經完成,于是便會調用XynCompleted方法,得到所需的ActionResult對象。

至于XyzCompleted方法所需要的參數,從代碼中便可看出是通過AsyncManager的Parameters集合進行“過渡”的。這里有個不是很理想的地方,便是使用了字符串這種“弱類型”的方式,假設參數名改變,則對應的字符串也需要跟著改變。

選擇Begin/End還是基于事件的異步模式?

很顯然,在ASP.NET MVC中使用既可以使用Begin/End或是基于事件的異步編程模式,因為ASP.NET MVC本身只是根據AsyncManager的行為來進行異步操作。不過在ASP.NET MVC中,似乎更看重的是基于事件的異步模式。我估計,這是由于兩種異步模式對于異常的行為差異所造成的吧。

正如我之前所提到的那樣,在使用Begin/End異步模式時,如果出現了錯誤則會在End方法調用時拋出異常。要知道在回調函數中拋出異常是異步編程中最危險的情況(有沒有之一?),如果沒有正確地進行捕獲則會讓整個進程崩潰——當然,我們也可以在配置文件中設置成“忽略”,但是這明顯也不妥當,例如會造成請求永遠無法結束,直至超時,并且有可能造成資源泄露。

與之相對,使用基于事件的異步模式則不會出現這個問題,因為在這種情況下,事件一定會被正確調用,而異常則永遠安安穩穩地保存在事件參數的Exception屬性中。因此,使用Begin/End則需要額外的try...catch進行保護,使用基于事件的異步編程模式則會讓代碼變得精簡一些。

當然,用著簡單,也只是因為那些異常已經被異步操作的“提供方”給處理了。試想,WebClient之所以可以通過事件參數來暴露異常,一定是因為在它內部使用了try...catch。同理,如果我們要實現一個基于事件的異步模式,例如上面的NewsService,那也一定少不了對異常進行仔細處理。

【編輯推薦】

  1. 詳細介紹Visual Studio 2010F#使用
  2. 對Visual Studio 2010F#代碼介紹
  3. TechED 09視頻專訪:F#與函數式編程語言
  4. 詳解F#異步及并行模式中的并行CPU及I/O計算
責任編輯:王曉東 來源: 老趙的博客
相關推薦

2010-04-07 16:51:59

F#

2009-07-22 10:13:31

異步ActionASP.NET MVC

2009-03-06 10:28:30

MVCASP.NET異步Action

2010-03-26 19:03:19

F#異步并行模式

2009-02-16 10:05:11

ActionMVCASP.NET

2009-02-17 09:22:14

ActionMVCASP.NET

2009-07-28 16:40:11

ASP.NET異步頁面

2012-03-31 10:59:02

ASP.NET

2013-04-01 15:25:41

異步編程異步EMP

2009-08-21 17:02:20

ASP.NET異步回調

2011-11-22 09:32:39

ASP.NET

2012-07-04 14:49:34

ASP.NET

2013-03-08 09:33:25

JavaScript同步異步

2010-03-26 18:31:59

F#異步并行模式

2010-03-16 09:09:04

F#

2009-08-21 17:11:15

ASP.NET異步回調

2009-07-29 17:29:46

ASP與ASP.NET

2009-07-22 09:11:02

Action方法ASP.NET MVC

2011-02-24 12:53:51

.NET異步傳統

2009-08-20 17:47:54

C#異步編程模式
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 久久久久久国产一区二区三区 | 精品久久久久久久久久久久久久久久久 | 亚洲天堂av在线 | 三级国产三级在线 | 午夜天堂精品久久久久 | 久久久亚洲一区 | 成人黄色电影在线播放 | 亚洲高清视频在线观看 | 精品视频在线一区 | 精品二区 | 午夜天堂精品久久久久 | 在线观看中文字幕亚洲 | 中日韩av| 免费看91 | www.99热.com | 中文字幕高清av | av在线免费观看网站 | 欧美激情精品久久久久久免费 | 日韩免费高清视频 | 在线看av的网址 | 精品综合久久 | 久久精品国产免费高清 | 99pao成人国产永久免费视频 | 欧美日韩一区二区三区四区 | 亚洲午夜视频在线观看 | 日韩另类 | 欧美中文字幕一区二区三区 | 成人美女免费网站视频 | 国产资源在线播放 | 国产福利91精品 | 国产美女精品视频免费观看 | 欧美日韩福利 | 天天干夜夜操 | 精品成人在线视频 | 成人免费久久 | 久久高清 | 亚洲国产精品久久久 | 国产激情精品 | 99久久久久久 | 亚洲美女在线视频 | 黄网站在线播放 |