ASP.NET MVC的非同步編程
在上篇《ASP.NET中的非同步編程》http://blog.csdn.net/zztfj/article/details/6837985的文章中,我們已經體會到非同步編程給網站帶來的高效能,高擴充性。現在ASP.NET已經發展到了MVC 3了,那麼在MVC中該如何進行非同步編程呢?放心,.NET FrameWork已經給我們提供了AsyncController(非同步控制器),用它可以實現MVC的非同步編程。下面關於它的使用作個簡單介紹。
要使用AsyncController進行非同步編程需要以下步驟。
1、 不要從 Controller 派生控制器,而應從AsyncController 派生。
2、 為Action建立兩個方法。這兩個方法分別以Async和Completed結尾。***Async方法返回void,***Completed方法返回ActionResult執行個體。儘管操作由兩個方法組成,但使用與同步操作方法相同的 URL 來訪問它(例如Portal/News?city=Seattle)。 其他方法(例如 RedirectToAction 和 RenderAction)還是將按照 News 而不是 NewsAsync 來引用操作方法。傳遞到 ***sAsync 的參數使用普通的參數綁定機制。傳遞到
***Completed 的參數使用 Parameters 字典。
使用AsyncController的注意點:
1、使用OutstandingOperations 屬性通知 ASP.NET 有多少個操作已掛起。 這是必要的,因為 ASP.NET 不能確定由操作方法啟動了多少個操作或這些操作何時完成。當OutstandingOperations 屬性為零時,ASP.NET 可通過調用 ***Completed 方法來完成整個非同步作業。
2、如果要將特性應用於非同步作業方法,則將它們應用於 ActionAsync 方法,而不是應用於ActionCompleted 方法。 忽略 ActionCompleted 方法上的特性。
已添加兩個新的特性:AsyncTimeoutAttribute 和 NoAsyncTimeoutAttribute。 這些特性可讓您控制非同步逾時時間。
3、如果非同步作業方法調用一個使用 BeginMethod/EndMethod 模式公開方法的服務,則回調方法(即作為非同步回調參數傳遞到 Begin 方法的方法)可能會在一個不由 ASP.NET 控制的線程上執行。 在此情況下,HttpContext.Current將為 null,並且當應用程式訪問 AsyncManager 類的成員(例如 Parameters)時可能會出現競爭條件。 若要確保已訪問HttpContext.Current 執行個體並避免競爭條件,則可以通過從回調方法中調用 Sync() 來還原
HttpContext.Current。
樣本一:在一個方法中調用三個基於事件的非同步WebService的樣本
private readonly ServiceClient _client = new ServiceClient(); public void AsynchronousAsync() { ViewData["Title"] = "Parallel asynchronous calls"; AsyncManager.OutstandingOperations.Increment(3); string city = "Seattle"; _client.GetHeadlinesCompleted += (sender, e) => { AsyncManager.Parameters["headlines"] = e.Result; AsyncManager.OutstandingOperations.Decrement(); }; _client.GetHeadlinesAsync(city); _client.GetScoresCompleted += (sender, e) => { AsyncManager.Parameters["scores"] = e.Result; AsyncManager.OutstandingOperations.Decrement(); }; _client.GetScoresAsync(city); _client.GetForecastCompleted += (sender, e) => { AsyncManager.Parameters["forecast"] = e.Result; AsyncManager.OutstandingOperations.Decrement(); }; _client.GetForecastAsync(city); } public ActionResult AsynchronousCompleted(NewsHeadline[] headlines, SportsScore[] scores, WeatherForecast[] forecast) { return View("Common", new ViewModel { Headlines = headlines, Scores = scores, Forecast = forecast }); }
完整的:http://archive.msdn.microsoft.com/aspnetmvcsamples/Release/ProjectReleases.aspx?ReleaseId=3547
樣本二:調用一個Begin/End模式的非同步WebService的樣本
public void NewsAsync(string city) { AsyncManager.OutstandingOperations.Increment(); _client.BeginGetHeadlines(city, ar => { if (ar.CompletedSynchronously) { AsyncManager.Parameters["news"] = _client.EndGetHeadlines(ar); AsyncManager.OutstandingOperations.Decrement(); } else { AsyncManager.Sync(() => { AsyncManager.Parameters["news"] = _client.EndGetHeadlines(ar); AsyncManager.OutstandingOperations.Decrement(); }); } }, null); } public ActionResult NewsCompleted(NewsHeadline[] headlines) { return View("Common", new ViewModel { Headlines = headlines }); }