技术开发 频道

使用ASP.NET MVC Futures 中的异步Action

  【IT168 技术文档】在ASP.NET MVC中使用异步是比较麻烦的,从RC1版开始ASP.NET MVC Futures中提供了几个支持异步的类。

  相关的类有:AsyncActionDescriptor、AsyncController、AsyncControllerActionInvoker、AsyncManager、AsyncResultWrapper、AsyncTimeoutAttribute、NoAsyncTimeoutAttribute。

  相关接口有:IAsyncActionDescriptor、IAsyncActionInvoker、IAsyncController、IAsyncManagerContainer。

  下面讲解一下怎么用它们

  一、使用异步Action前的准备工作

  1.引用Microsoft.Web.Mvc。    

  2.先要将要异步处理的Url交由MvcHttpAsyncHandler处理,这一步可以由AsyncRouteCollectionExtensions.MapAsyncRoute来设置规则,将原MapRoute处理的规则改为MapAsyncRoute,如:

routes.MapAsyncRoute(
                
"Default",
                
"{controller}/{action}/{id}",
                
new { controller = "Home", action = "Index", id = "" }
            );

 

  3.将相应的Controller继承于AsyncController。

public class HomeController : AsyncController    {    }

  4.我们约定以下定义的Action都在HomeController中

  二、第一种异步Action方式:Action、ActionCompleted方式

ASP.NET MVC Futures支持按名称自动寻找异步Action的方法

  其使用方法为:

public void Async1(){
            
//主线程
        }
        
public ActionResult Async1Completed(){
            
//自动寻找与主线程 Action名称+Completed 的Action 做为异步Action
            return Content("Async1");}

  三、第二种异步Action方式:BeginAction、EndAction方式

  如果第一种方式你了解了的话第二种自然也不在话下,不过这种方式是与其它类的异步调用一起使用。

  public delegate void AsyncEventHandler();//这里声明了一个委托,//也可使用WebRequest/WebResponse/SqlConnection来实现这个异步过程

  public void Event1(){}
        
public IAsyncResult BeginAsync3(AsyncCallback callback, object state){
           AsyncEventHandler asy
= new AsyncEventHandler(Event1);
            
  ViewData[
"a"]=asy;//这里在方法間传值必须使用辅助存储对象,第一种方法中也是一样
            return asy.BeginInvoke(callback, state);
        }
        
public void EndAsync3(IAsyncResult result){
//转到异步的Action中
            var a = ViewData["a"] as AsyncEventHandler;
            a.EndInvoke(result);
            Content(
"完成").ExecuteResult(this.ControllerContext);
        }

   四、第三种异步Action方式:使用AsyncManager.RegisterTask及委托

  如果感觉上面使用2个方法才能实现异步Action有些麻烦的话(也的确是麻烦),可以使用AsyncManager.RegisterTask来调用委托来实现异步调用。

public void  Async2(){
            
this.AsyncManager.RegisterTask(c => {
                
//主线程,调用异步线程
                c(null);
            },
delegate(IAsyncResult result) {
                
//异步部分              
                Content("Async2").ExecuteResult(this.ControllerContext);
            });
        }

 

  其实无论是哪种方法都感觉不太完美,我个人觉得Action/ActionCompleted的方法可能更优美一点,适合一般使用(只是这三个比较)。只是AsyncManager.RegisterTask的方法传值方便一点,而Begin/End方法更适合与其它异步的操作配合。

0
相关文章