精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

ASP.NET MVC涉及到的5個同步與異步

開發(fā) 后端
在ASP.NETMVC的整個體系中涉及到很多同步/異步的執(zhí)行方式,雖然在前面相應(yīng)的文章中已經(jīng)對此作了相應(yīng)的介紹,為了讓讀者對此有一個整體的了解,我們來做一個總結(jié)性的論述。

Action方法的執(zhí)行具有兩種基本的形式,即同步執(zhí)行和異步執(zhí)行,而在ASP.NETMVC的整個體系中涉及到很多同步/異步的執(zhí)行方式,雖然在前面相應(yīng)的文章中已經(jīng)對此作了相應(yīng)的介紹,為了讓讀者對此有一個整體的了解,我們來做一個總結(jié)性的論述。

一、MvcHandler的同步與異步

對于ASP.NET MVC應(yīng)用來說,MvcHandler是最終用于處理請求的HttpHandler,它是通過UrlRoutingModule這個實現(xiàn)了URL路由的HttpModule被動態(tài)映射到相應(yīng)的請求的。MvcHandler借助于ControllerFactory激活并執(zhí)行目標(biāo)Controller,并在執(zhí)行結(jié)束后負(fù)責(zé)對激活的Controller進(jìn)行釋放,相關(guān)的內(nèi)容請參與本書的第3章“Controller的激活”。如下面的代碼片斷所示,MvcHandler同時實現(xiàn)了IHttpHandler和IHttpAsyncHandler接口,所以它總是調(diào)用BeginProcessRequest/EndProcessRequest方法以異步的方式來處理請求。

 

  1. public class MvcHandler : IHttpAsyncHandler, IHttpHandler, ...  
  2. {  
  3.     //其他成員     
  4.     IAsyncResult IHttpAsyncHandler.BeginProcessRequest(HttpContext context, AsyncCallback cb, object extraData);  
  5.     void IHttpAsyncHandler.EndProcessRequest(IAsyncResult result);  
  6.     void IHttpHandler.ProcessRequest(HttpContext httpContext);  

#p#

二、Controller的同步與異步

Controller也具有同步與異步兩個版本,它們分別實現(xiàn)了具有如下定義的兩個接口IController和IAsyncController。當(dāng)激活的Controller對象在MvcHandler的BeginProcessRequest方法中是按照這樣的方式執(zhí)行的:如果Controller的類型實現(xiàn)了IAsyncController接口,則調(diào)用BeginExecute/EndExecute方法以異步的方式執(zhí)行Controller;否則Controller的執(zhí)行通過調(diào)用Execute方法以同步方式執(zhí)行。

  1. public interface IController  
  2. {      
  3.     void Execute(RequestContext requestContext);  
  4. }  
  5. public interface IAsyncController : IController  
  6. {      
  7.     IAsyncResult BeginExecute(RequestContext requestContext, AsyncCallback callback, object state);  
  8.     void EndExecute(IAsyncResult asyncResult);  

默認(rèn)情況下通過Visual Studio的向?qū)?chuàng)建的Controller類型是抽象類型Controller的子類。如下面的代碼片斷所示,Controller同時實現(xiàn)了IController和IAsyncController這兩個接口,所以當(dāng)MvcHandler進(jìn)行請求處理時總是以異步的方式來執(zhí)行Controller。

  1. public abstract class Controller : ControllerBase, IController, IAsyncController, ...  
  2. {  
  3.     //其他成員  
  4.     protected virtual bool DisableAsyncSupport  
  5.     {  
  6.         get{return false;}  
  7.     }  

但是Controller類型具有一個受保護(hù)的只讀屬性DisableAsyncSupport用于表示是否禁用對異步執(zhí)行的支持。在默認(rèn)情況下,該屬性值為False,所以默認(rèn)情況下是支持Controller的異步執(zhí)行的。如果我們通過重寫該屬性將值設(shè)置為True,那么Controller將只能以同步的方式執(zhí)行。具體的實現(xiàn)邏輯體現(xiàn)在如下的代碼片斷中:BeginExecute方法在DisableAsyncSupport屬性為True的情況下通過調(diào)用Execute方法(該方法會調(diào)用一個受保護(hù)的虛方法ExecuteCore最終對Controller進(jìn)行同步執(zhí)行);否則通過調(diào)用BeginExecuteCore/EndExecuteCore以異步方式執(zhí)行Controller。

  1.  public abstract class Controller: ...  
  2.  {  
  3.      //其他成員  
  4.      protected virtual IAsyncResult BeginExecute(RequestContext requestContext,   
  5.      AsyncCallback callback, object state)  
  6.      {  
  7.          if (this.DisableAsyncSupport)  
  8.          {  
  9.              //通過調(diào)用Execute方法同步執(zhí)行Controller  
  10.         }  
  11.         else 
  12.         {  
  13.             //通過調(diào)用BeginExecuteCore/EndExecuteCore方法異步執(zhí)行Controller  
  14.         }  
  15. }  
  16.     protected override void ExecuteCore();  
  17.     protected virtual IAsyncResult BeginExecuteCore(AsyncCallback callback, object state);  
  18.     protected virtual void EndExecuteCore(IAsyncResult asyncResult);  

#p#

三、 ActionInvoker的同步與異步

包括Model綁定與驗證的整個Action的執(zhí)行通過一個名為ActionInvoker的組件來完成,而它同樣具有同步和異步兩個版本,分別實現(xiàn)了接口IActionInvoker和IAsyncActionInvoker。如下面的代碼片斷所示,這兩個接口分別通過InvokeAction和BeginInvokeAction/EndInvokeAction方法以同步和異步的方式執(zhí)行Action。抽象類Controller中具有一個ActionInvoker屬性用于設(shè)置和返回用于執(zhí)行自身Action的ActionInvoker對象,而該對象最終是通過受保護(hù)需方法CreateActionInvoker創(chuàng)建的。

  1.  public interface IActionInvoker  
  2.  {  
  3.      bool InvokeAction(ControllerContext controllerContext, string actionName);  
  4.  }  
  5.     
  6.  public interface IAsyncActionInvoker : IActionInvoker  
  7.  {  
  8.      IAsyncResult BeginInvokeAction(ControllerContext controllerContext, string actionName, AsyncCallback callback, object state);  
  9.      bool EndInvokeAction(IAsyncResult asyncResult);  
  10. }  
  11.    
  12. public abstract class Controller  
  13. {     
  14.     //其它成員  
  15.     public IActionInvoker ActionInvoker { get; set; }  
  16.     protected virtual IActionInvoker CreateActionInvoker()  

ASP.NET MVC真正用于Action方法同步和異步執(zhí)行的ActionInvoker分別是ControllerActionInvoker和AsyncControllerActionInvoker。如下面的代碼片斷所示,ControllerActionInvoker定義了一個受保護(hù)的方法GetControllerDescriptor用于根據(jù)指定的Controller上下文獲取相應(yīng)的ControllerDescriptor,它的子類AsyncControllerActionInvoker對這個方法進(jìn)行了重寫。

  1. public class ControllerActionInvoker : IActionInvoker  
  2. {  
  3.     //其它成員  
  4.     protected virtual ControllerDescriptor GetControllerDescriptor(ControllerContext controllerContext);  
  5. }  
  6.    
  7. public class AsyncControllerActionInvoker : ControllerActionInvoker,IAsyncActionInvoker, IActionInvoker  
  8. {  
  9.     //其它成員  
  10.   protected override ControllerDescriptor GetControllerDescriptor(ControllerContext controllerContext);  

我們所有要了解的是在默認(rèn)情況下(沒有對Controller類型的ActionInvoker屬性進(jìn)行顯式設(shè)置)采用的ActionInvoker類型是哪個。ASP.NET MVC對Conroller采用的ActionInvoker類型的選擇機(jī)制是這樣的:

通過當(dāng)前的DependencyResolver以IAsyncActionInvoker接口去獲取注冊的ActionInvoker,如果返回對象不為Null,則將其作為默認(rèn)的ActionInvoker。

通過當(dāng)前的DependencyResolver以IActionInvoker接口去獲取注冊的ActionInvoker,如果返回對象不為Null,則將其作為默認(rèn)的ActionInvoker。

創(chuàng)建AsyncControllerActionInvoker對象作為默認(rèn)的ActionInvoker。

在默認(rèn)的情況下,當(dāng)前的DependencyResolver直接通過對指定的類型進(jìn)行反射來提供對應(yīng)的實例對象,所以對于前面兩個步驟返回的對象均為Null,所以默認(rèn)創(chuàng)建出來的ActionInvoker類型為AsyncControllerActionInvoker。我們可以通過如下一個簡單的實例來驗證這一點。在通過Visual Studio的ASP.NET MVC項目模板創(chuàng)建的空Web應(yīng)用中,我們創(chuàng)建了如下一個默認(rèn)的HomeController,在Action方法Index中直接通過ContentResult將ActionInvoker屬性的類型名稱呈現(xiàn)出來。

  1.  public class HomeController : Controller  
  2.  {    
  3.      public ActionResult Index()  
  4.      {  
  5.          return Content("默認(rèn)ActionInvoker類型:" + this.ActionInvoker.GetType().FullName);  
  6.      }  

當(dāng)運行該Web應(yīng)用時,會在瀏覽器上產(chǎn)生如下的輸出結(jié)果,我們可以清楚地看到默認(rèn)采用的ActionInvoker類型正是AsyncControllerActionInvoker。

  1. 默認(rèn)ActionInvoker類型:System.Web.Mvc.Async.AsyncControllerActionInvoker 

為了進(jìn)一步驗證基于DependencyResolver對ActionInvoker的提供機(jī)制,我們將《ASP.NET MVC Controller激活系統(tǒng)詳解:IoC的應(yīng)用》創(chuàng)建的基于Ninject的自定義NinjectDependencyResolver應(yīng)用在這里。如下面的代碼片斷所示,在初始化NinjectDependencyResolver的時候,我們將IActionInvoker和IAsyncActionInvoker影射到兩個自定義ActionInvoker類型,即FooActionInvoker和FooAsyncActionInvoker,它們分別繼承自ControllerActionInvoker和AsyncControllerActionInvoker。

  1.  public class NinjectDependencyResolver : IDependencyResolver  
  2.  {  
  3.      public IKernel Kernel { get; private set; }  
  4.      public NinjectDependencyResolver()  
  5.      {  
  6.          this.Kernel = new StandardKernel();  
  7.          AddBindings();  
  8.      }  
  9.      private void AddBindings()  
  10.     {  
  11.         this.Kernel.Bind<IActionInvoker>().To<FooActionInvoker>();  
  12.         this.Kernel.Bind<IAsyncActionInvoker>().To<FooAsyncActionInvoker>();  
  13.     }  
  14.     public object GetService(Type serviceType)  
  15.     {  
  16.         return this.Kernel.TryGet(serviceType);  
  17.     }  
  18.     public IEnumerable<object> GetServices(Type serviceType)  
  19.     {  
  20.         return this.Kernel.GetAll(serviceType);  
  21.     }  
  22. }  
  23. public class FooActionInvoker : ControllerActionInvoker  
  24. {}  
  25. public class FooAsyncActionInvoker : AsyncControllerActionInvoker  
  26. {} 

在Global.asax中對NinjectDependencyResolver進(jìn)行注冊后運行我們的程序,會在瀏覽器中得到如下的輸出結(jié)果。IAsyncActionInvoker和FooAsyncActionInvoker進(jìn)行了影射,NinjectDependencyResolver可以通過IAsyncActionInvoker提供一個FooAsyncActionInvoker實例。

  1. 默認(rèn)ActionInvoker類型:Artech.Mvc.FooAsyncActionInvoker 

現(xiàn)在我們對NinjectDependencyResolver的定義稍加修改,將針對IAsyncActionInvoker接口的類型影射刪除,只保留針對IActionInvoker的映射。

  1. public class NinjectDependencyResolver : IDependencyResolver  
  2. {  
  3.     //其它成員  
  4.     private void AddBindings()  
  5.     {  
  6.         this.Kernel.Bind<IActionInvoker>().To<FooActionInvoker>();  
  7.         //this.Kernel.Bind<IAsyncActionInvoker>().To<FooAsyncActionInvoker>();  
  8.     }  

再次運行我們的程序則會得到如下的輸出結(jié)果。由于NinjectDependencyResolver只能通過IActionInvoker接口提供具體的ActionInvoker,所以最終被創(chuàng)建的是一個FooActionInvoker對象。這個實例演示告訴我們:當(dāng)我們需要使用到自定義的ActionInvoker的時候,可以通過自定義DependencyResolver以IoC的方式提供具體的ActionInvoker實例。

  1. 默認(rèn)ActionInvoker類型:Artech.Mvc.FooActionInvoker 

四、ControllerDescriptor的同步與異步

如果采用ControllerActionInvoker,Action總是以同步的方式來直接,但是當(dāng)AsyncControllerActionInvoker作為Controller的ActionInvoker時,并不意味著總是以異步的方式來執(zhí)行所有的Action。至于這兩種類型的ActionInvoker具體采用對Action的怎樣的執(zhí)行方式,又涉及到兩個描述對象,即用于描述Controller和Action的ControllerDescriptor和ActionDescriptor。

通過前面“Model的綁定”中對這兩個對象進(jìn)行過相應(yīng)的介紹,我們知道在ASP.NET MVC應(yīng)用編程接口中具有兩個具體的ControllerDescriptor,即ReflectedControllerDescriptor和ReflectedAsyncControllerDescriptor,它們分別代表同步和異步版本的ControllerDescriptor。

  1. public class ReflectedControllerDescriptor : ControllerDescriptor  
  2. {  
  3.     //省略成員  
  4. }  
  5.    
  6. public class ReflectedAsyncControllerDescriptor : ControllerDescriptor  
  7. {  
  8.     //省略成員  

ReflectedControllerDescriptor和ReflectedAsyncControllerDescriptor并非對分別實現(xiàn)了IController和IAyncController接口的Controller的描述,而是對直接繼承自抽象類Controller和AsyncController的Controller的描述。它們之間的區(qū)別在于創(chuàng)建者的不同,在默認(rèn)情況下ReflectedControllerDescriptor和ReflectedAsyncControllerDescriptor分別是通過ControllerActionInvoker和AsyncControllerActionInvoker來創(chuàng)建的。ActionInvoker和ControllerDescriptor之間的關(guān)系可以通過如下圖所示的UML來表示。

image

ActionInvoker與ControllerDescriptor之間的關(guān)系可以通過一個簡單的實例來驗證。在通過Visual Studio的ASP.NET MVC項目模板創(chuàng)建的空Web應(yīng)用中,我們自定義了如下兩個分別繼承自ControllerActionInvoker和AsyncControllerActionInvoker的ActionInvoker類型。在這兩個自定義ActionInvoker中,定義了公有的GetControllerDescriptor方法覆蓋了基類的同名方法(受保護(hù)的虛方法),并直接直接調(diào)用基類的同名方法根據(jù)提供的Controller上下文的到相應(yīng)的ControllerDescriptor對象。

  1.  public class FooActionInvoker : ControllerActionInvoker  
  2.  {   
  3.      public new ControllerDescriptor GetControllerDescriptor(ControllerContext controllerContext)  
  4.      {  
  5.          return base.GetControllerDescriptor(controllerContext);  
  6.      }  
  7.  }  
  8.     
  9.  public class BarActionInvoker : AsyncControllerActionInvoker  
  10. {  
  11.     public new ControllerDescriptor GetControllerDescriptor(ControllerContext controllerContext)  
  12.     {  
  13.         return base.GetControllerDescriptor(controllerContext);  
  14.     }  

然后我們定義了兩個Controller類型,它們均是抽象類型Controller的直接繼承者。如下面的代碼片斷所示,這兩Controller類(FooController和BarController)都重寫了虛方法CreateActionInvoker,而返回的ActionInvoker類型分別是上面我們定義的FooActionInvoker和BarActionInvoker。在默認(rèn)的Action方法Index中,我們利用當(dāng)前的ActionInvoker得到用于描述本Controller的ControllerDescriptor對象,并將其類型呈現(xiàn)出來。

  1.  public class FooController : Controller  
  2.  {  
  3.      protected override IActionInvoker CreateActionInvoker()  
  4.      {  
  5.          return new FooActionInvoker();  
  6.      }  
  7.     
  8.      public void Index()  
  9.      {  
  10.         ControllerDescriptor controllerDescriptor = ((FooActionInvoker)this.ActionInvoker).GetControllerDescriptor(ControllerContext);  
  11.         Response.Write(controllerDescriptor.GetType().FullName);  
  12.     }  
  13. }  
  14.    
  15. public class BarController : Controller  
  16. {  
  17.     protected override IActionInvoker CreateActionInvoker()  
  18.     {  
  19.         return new BarActionInvoker();  
  20.     }  
  21.    
  22.     public void Index()  
  23.     {  
  24.         ControllerDescriptor controllerDescriptor = ((BarActionInvoker)this.ActionInvoker).GetControllerDescriptor(ControllerContext);  
  25.         Response.Write(controllerDescriptor.GetType().FullName);  
  26.     }  

現(xiàn)在我們運行我們的程序,并在瀏覽器中輸入相應(yīng)的地址對定義在FooController和BarController的默認(rèn)Action方法Index發(fā)起訪問,相應(yīng)的ControllerDescriptor類型名稱會以下圖所示的形式呈現(xiàn)出來,它們的類型分別是ReflectedControllerDescriptorReflectedAsyncControllerDescriptor

image

#p#

五、ActionDescriptor的執(zhí)行

Controller包含一組用于描述Action方法的ActionDescriptor對象。由于Action方法可以采用同步和異步執(zhí)行方法,異步Action對應(yīng)的ActionDescriptor直接或者間接繼承自抽象類AsyncActionDescriptor,后者是抽象類ActionDescriptor的子類。如下面的代碼片斷所示,同步和異步Action的執(zhí)行分別通過調(diào)用Execute和BeginExecute/EndExecute方法來完成。值得一提的是,AsyncActionDescriptor重寫了Execute方法并直接在此方法中拋出一個InvalidOperationException異常,也就是說AsyncActionDescriptor對象只能采用 異步執(zhí)行方式。

  1.  public abstract class ActionDescriptor : ICustomAttributeProvider  
  2.  {  
  3.      //其他成員  
  4.      public abstract object Execute(ControllerContext controllerContext, IDictionary<string, object> parameters);  
  5.  }  
  6.     
  7.  public abstract class AsyncActionDescriptor : ActionDescriptor  
  8.  {  
  9.      //其他成員  
  10.     public abstract IAsyncResult BeginExecute(ControllerContext controllerContext, IDictionary<string, object> parameters, AsyncCallback callback, object state);  
  11.     public abstract object EndExecute(IAsyncResult asyncResult);  

通過前面“Model的綁定”我們知道,在ASP.NET MVC應(yīng)用編程接口中采用ReflectedControllerDescriptor來描述同步Action。異步Action方法具有兩種不同的定義方式:其一,通過兩個匹配的方法XxxAsync/XxxCompleted定義;其二,通過返回類型為Task的方法來定義。這兩種異步Action方法對應(yīng)的AsyncActionDescriptor類型分別是ReflectedAsyncActionDescriptor和TaskAsyncActionDescriptor

對于ReflectedControllerDescriptor來說,包含其中的ActionDescriptor類型均為ReflectedActionDescriptor。而ReflectedAsyncControllerDescriptor描述的Controller可以同時包含同步和異步的Action方法,ActionDescriptor的類型取決于Action方法定義的方式。ControllerDescriptor與ActionDescriptor之間的關(guān)系如下圖所示的UML來表示。

image

#p#

實例演示:AsyncActionInvoker對ControllerDescriptor的創(chuàng)建

為了讓讀者對ActionInvoker對ControllerDescriptor的解析機(jī)制具有一個深刻的理解,同時也作為對該機(jī)制的驗證,我們做一個簡單的實例演示。通過前面的介紹我們知道在默認(rèn)的情況下Controller采用AsyncControllerActionInvoker進(jìn)行Action方法的執(zhí)行,這個例子就來演示一下它生成的ControllerDescriptor是個怎樣的對象。我們通過Visual Studio的ASP.NET MVC項目模板創(chuàng)建一個空Web應(yīng)用,并創(chuàng)建一個默認(rèn)的HomeController,然后對其進(jìn)行如下的修改。

  1. public class HomeController : AsyncController  
  2. {  
  3.     public void Index()  
  4.     {  
  5.         MethodInfo method = typeof(AsyncControllerActionInvoker).GetMethod("GetControllerDescriptor", BindingFlags.Instance | BindingFlags.NonPublic);  
  6.         ControllerDescriptor controllerDescriptor = (ControllerDescriptor)method.Invoke(this.ActionInvoker, new object[] { this.ControllerContext });  
  7.         Response.Write(controllerDescriptor.GetType().FullName + "<br/>");  
  8.         CheckAction(controllerDescriptor, "Foo");  
  9.         CheckAction(controllerDescriptor, "Bar");  
  10.        CheckAction(controllerDescriptor, "Baz");  
  11.  
  12.    }  
  13.     private void CheckAction(ControllerDescriptor controllerDescriptor,string actionName)  
  14.    {  
  15.        ActionDescriptor actionDescriptor = controllerDescriptor.FindAction(this.ControllerContext, actionName);  
  16.        Response.Write(string.Format("{0}: {1}<br/>",actionName,actionDescriptor.GetType().FullName));  
  17.    }  
  18.  
  19.    public void Foo() { }  
  20.    public void BarAsync() { }  
  21.    public void BarCompleted() { }  
  22.    public Task<ActionResult> Baz()  
  23.    {  
  24.        throw new NotImplementedException();  
  25.    }      

我們首先將HomeController的基類從Controller改為AsyncController,并定義了Foo、BarAsync/BarCompleted和Baz四個方法,我們知道它們對應(yīng)著Foo、Bar和Baz三個Action,其中Foo是同步Action,Bar和Baz分別是兩種不同定義形式(XxxAsync/XxxCompleted和Task)的異步Action。

CheckAction用于根據(jù)指定的Action名稱從ControllerDescriptor對象中獲取用于表示對應(yīng)Action的ActionDescriptor對象,最終將類型名稱呈現(xiàn)出來。在Index方法中,我們通過反射的方式調(diào)用當(dāng)前ActionInvoker(一個AsyncControllerActionInvoker對象)的受保護(hù)方法GetControllerDescriptor或者用于描述當(dāng)前Controller(HomeController)的ControllerDescriptor的對象,并將類型名稱呈現(xiàn)出來。最后通過調(diào)用CheckAction方法將包含在創(chuàng)建的ControllerDescriptor對象的三個ActionDescriptor類型呈現(xiàn)出來。

當(dāng)我們運行該程序的時候,在瀏覽器中會產(chǎn)生如下的輸出結(jié)果,從中可以看出ControllerDescriptor類型為ReflectedAsyncControllerDescriptor。同步方法Foo對象的ActionDescriptor是一個ReflectedActionDescriptor對象;以XxxAsync/XxxCompleted形式定義的異步方法Bar對應(yīng)的ActionDescriptor是一個ReflectedAsyncActionDescriptor對象;而返回類型為Task的方法Baz對應(yīng)的ActionDescriptor類型則是TaskAsyncActionDescriptor。

image

AsyncController、AsyncControllerActionInvoker與AsyncActionDescriptor

不論我們采用哪種形式的定義方式,步Action方法都只能定義在繼承自AsyncController的Controller類型中,否則將被認(rèn)為是同步方法。此外,由于通過ControllerActionInvoker只能創(chuàng)建包含ReflectedActionDescriptor的ReflectedControllerDescriptor,如果我們在AsyncController中采用ControllerActionInvoker對象作為ActionInvoker,所有的Action方法也將被認(rèn)為是同步的。

我們同樣可以采用一個簡單的實例演示來證實這一點。在通過Visual Studio的ASP.NET MVC項目模板創(chuàng)建的空Web應(yīng)用中我們定義如下三個Controller(FooController、BarController和BazController)。我們重寫了它們的CreateActionInvoker方法,返回的ActionInvoker類型(FooActionInvoker和BarActionInvoker)定義在上面,在這里我們將FooActionInvoker和BarActionInvoker看成是ControllerActionInvoker和AsyncControllerActionInvoker(默認(rèn)使用的ActionInvoker)。

  1.  public class FooController : AsyncController  
  2.  {  
  3.      protected override IActionInvoker CreateActionInvoker()  
  4.      {  
  5.          return new BarActionInvoker();  
  6.      }  
  7.      public void Index()  
  8.      {  
  9.          BarActionInvoker actionInvoker = (BarActionInvoker)this.ActionInvoker;  
  10.         Utility.DisplayActions(controllerContext=>actionInvoker.GetControllerDescriptor(ControllerContext),ControllerContext);  
  11.     }  
  12.    
  13.     public void DoSomethingAsync()  
  14.     { }  
  15.     public void DoSomethingCompleted()  
  16.     { }  
  17. }  
  18.    
  19. public class BarController : Controller  
  20. {  
  21.     protected override IActionInvoker CreateActionInvoker()  
  22.     {  
  23.         return new BarActionInvoker();  
  24.     }  
  25.     public void Index()  
  26.     {  
  27.         BarActionInvoker actionInvoker = (BarActionInvoker)this.ActionInvoker;  
  28.         Utility.DisplayActions(controllerContext => actionInvoker.GetControllerDescriptor(ControllerContext), ControllerContext);  
  29.     }  
  30.     public void DoSomethingAsync()  
  31.     { }  
  32.     public void DoSomethingCompleted()  
  33.     { }  
  34. }  
  35.    
  36. public class BazController : Controller  
  37. {  
  38.     protected override IActionInvoker CreateActionInvoker()  
  39.     {  
  40.         return new FooActionInvoker();  
  41.     }  
  42.     public void Index()  
  43.     {  
  44.         FooActionInvoker actionInvoker = (FooActionInvoker)this.ActionInvoker;  
  45.         Utility.DisplayActions(controllerContext => actionInvoker.GetControllerDescriptor(ControllerContext), ControllerContext);  
  46.     }  
  47.     public void DoSomethingAsync()  
  48.     { }  
  49.     public void DoSomethingCompleted()  
  50.     { }  
  51. }  
  52.    
  53. public static class Utility  
  54. {  
  55.     public static void DisplayActions(Func<ControllerContext, ControllerDescriptor> controllerDescriptorAccessor, ControllerContext controllerContext)  
  56.     {   
  57.         ControllerDescriptor controllerDescriptor = controllerDescriptorAccessor(controllerContext);  
  58.         string[] actionNames = { "DoSomething""DoSomethingAsync""DoSomethingCompleted" };  
  59.         Array.ForEach(actionNames, actionName =>  
  60.             {  
  61.                 ActionDescriptor actionDescriptor = controllerDescriptor.FindAction(controllerContext,actionName);  
  62.                 if (null != actionDescriptor)  
  63.                 {  
  64.                     HttpContext.Current.Response.Write(string.Format("{0}: {1}<br/>", actionDescriptor.ActionName, actionDescriptor.GetType().Name));  
  65.                 }  
  66.             });  
  67.     }  

我們在三個Controller中以異步Action的形式定義了兩個方法DoSomethingAsync和DoSomethingCompleted。FooController繼承自AsyncController,使用AsyncControllerActionInvoker作為其ActionInvoker,這是正常的定義;BarController雖然采用AsyncControllerActionInvoker,但是將抽象類Controller作為其基類;而BazController雖然繼承自ActionInvoker,但ActionInvoker類型為ControllerActionInvoker。在默認(rèn)的Action方法Index中,我們將通過DoSomethingAsync和DoSomethingCompleted方法定義的Action的名稱和對應(yīng)的ActionDescriptor類型輸出來。

如果我們運行該程序,并在瀏覽器中輸入相應(yīng)的地址對定義在三個Controller的默認(rèn)Action方法Index發(fā)起訪問,會呈現(xiàn)出如下圖所示的結(jié)果。我們可以清楚地看到,對于以XxxAsync/XxxCompleted形式定義的“異步”Action方法定義,只有針對AsyncController并且采用AsyncControllerActionInvoker的情況下才會被解析為一個異步Action。如果不滿足這兩個條件,它們會被視為兩個普通的同步Action。

image

Action方法的執(zhí)行

目標(biāo)Action方法的最終執(zhí)行由被激活的Controller的ActionInvoker決定,ActionInvoker最終通過調(diào)用對應(yīng)的ActionDescriptor來執(zhí)行被它描述的Action方法。如果采用ControllerActionInvoker,被它創(chuàng)建的ControllerDescriptor(ReflectedControllerDescriptor)只包含同步的ActionDescriptor(ReflectedActionDescriptor),所以Action方法總是以同步的方式被執(zhí)行。

如果目標(biāo)Controller是抽象類Controller的直接繼承者,這也是通過Visual Studio的Controller創(chuàng)建向?qū)У哪J(rèn)定義方式,ActionInvoker(ControllerActionInvoker/AsyncControllerActionInvoker)的選擇只決定了創(chuàng)建的ControllerDescriptor的類型(ReflectedControllerDescriptor/ReflectedAsyncControllerDescriptor),ControllerDescriptor包含的所有ActionDescriptor依然是同步的(ReflectedActionDescriptor),所以Action方法也總是以同步的方式被執(zhí)行。

以異步方式定義的Action方法(XxxAsync/XxxCompleted或采用Task返回類型)只有定義在繼承自AsyncController的Controller類型中,并且采用AsyncControllerActionInvoker作為其ActionInvoker,最終才會創(chuàng)建AsyncActionDescriptor來描述該Action。也只有同時滿足這兩個條件,Action方法才能以異步的方式執(zhí)行。

原文鏈接:http://www.cnblogs.com/artech/archive/2012/06/22/sync-and-async-02.html

【編輯推薦】

  1. ASP.NET Cache的一些總結(jié)
  2. ASP.NET Page那點事
  3. 逃脫Asp.Net MVC框架枷鎖 使用Razor視圖引擎
  4. ASP.NET MVC3 從零開始一步步構(gòu)建Web
  5. 用ASP.NET MVC 3實現(xiàn)一個訪問統(tǒng)計系統(tǒng)
責(zé)任編輯:張偉 來源: Artech的博客
相關(guān)推薦

2009-07-22 10:13:31

異步ActionASP.NET MVC

2015-07-03 10:12:04

編程同步非阻塞

2009-03-06 10:28:30

MVCASP.NET異步Action

2012-02-03 09:06:53

ASP.NET MVC

2009-07-31 12:43:59

ASP.NET MVC

2009-07-24 13:20:44

MVC框架ASP.NET

2009-12-08 08:57:21

ASP.NET MVC

2009-02-17 09:22:14

ActionMVCASP.NET

2009-02-16 10:05:11

ActionMVCASP.NET

2009-03-13 10:58:48

ASP.NetMVC框架編程

2009-08-24 09:18:34

ASP.NET MVC

2015-06-29 10:00:02

ASP.NETMVC6

2014-08-26 09:22:40

ASP.NET MVCRouting

2009-07-20 15:44:32

ASP.NET MVC

2009-07-22 09:11:02

Action方法ASP.NET MVC

2009-07-22 10:34:37

ActionInvokASP.NET MVC

2009-07-22 10:09:59

ASP.NET MVC

2009-07-23 15:44:39

ASP.NET MVC

2009-07-20 10:53:59

ASP.NET MVC

2009-07-22 13:24:24

ASP.NET MVC
點贊
收藏

51CTO技術(shù)棧公眾號

天天操综合网| 91精品久久久久久久久久入口| 国产在线高清精品| 亚洲av成人片无码| 亚洲欧美自拍视频| 日本在线一区二区三区| 精品久久香蕉国产线看观看亚洲| 成人aa视频在线观看| 欧美成人milf| 亚洲欧美另类久久久精品2019| 欧美高跟鞋交xxxxhd| 天天影视综合色| 欧美日韩一区在线播放| x88av在线| 草草在线视频| 偷拍一区二区| 亚洲永久免费视频| 国产精品影片在线观看| 久久一区二区电影| 蜜臀av国内免费精品久久久夜夜| 亚洲电影一级片| 亚洲丝袜美腿综合| 国产精品一区二区三| 国产成人精品免高潮在线观看| 成人黄色片视频| 四虎在线视频免费观看| av免费播放网站| a在线观看视频| 婷婷丁香综合| 亚洲精品一区二区三区福利| 黄色网在线视频| 国产视频第二页| 小说区亚洲自拍另类图片专区 | 欧美三级视频在线| 欧美精品二区三区四区免费看视频| 91tv国产成人福利| 欧美一二区在线观看| 日本精品一区二区三区四区的功能| 欧美伦理一区二区| 亚洲成人第一区| 精品动漫3d一区二区三区免费版 | 成人免费网站在线观看| 天码人妻一区二区三区在线看| 老司机凹凸av亚洲导航| 欧美日韩国产区| 久久久久久高清| 在线观看日本视频| 激情91久久| 亚洲男人天堂2019| 久久久久国产一区| 日本高清视频在线观看| 国产一区二区精品在线观看| 欧美激情奇米色| 国产色无码精品视频国产| 秋霞一区二区| 欧美欧美欧美欧美| 婷婷无套内射影院| 成年人视频免费在线观看| 蜜桃av一区二区在线观看| 久久久精品电影| 无码国产69精品久久久久网站| 中文在线资源| 成人免费在线播放视频| 国产欧美日韩视频一区二区三区| 无码人妻熟妇av又粗又大| 日韩欧美一区二区三区免费看| 日韩色视频在线观看| 18禁男女爽爽爽午夜网站免费| 2021中文字幕在线| 国产精品久久久久aaaa| 国产精品手机视频| 这里只有精品9| 日韩视频在线一区二区三区 | www.色天使| 欧美亚洲综合视频| 亚洲第一在线综合网站| 深夜福利成人| 人人妻人人澡人人爽人人欧美一区| 日韩中文字幕区一区有砖一区 | 亚洲小说区图片| 日韩理论片av| 中文字幕久久久| 男人操女人下面视频| 综合久久2023| 夜夜揉揉日日人人青青一国产精品| 国产又粗又大又爽的视频| 精品美女视频在线观看免费软件| 国产一区二区三区黄视频| 97免费资源站| 性色av一区二区三区四区| 在线电影一区| 欧美在线视频一区| 国产亚洲欧美精品久久久久久| 成人精品久久| 欧美成人网在线| 亚洲欧美日韩第一页| 欧美天堂影院| 欧美tk丨vk视频| 亚洲欧美一区二区三区不卡| 加勒比久久高清| 综合欧美国产视频二区| 国产主播在线播放| 国产综合自拍| 国产不卡在线观看| av免费在线不卡| 久久夜色精品一区| 欧美不卡1区2区3区| 色窝窝无码一区二区三区成人网站 | 国产色产综合色产在线视频| 国产精品久久精品国产| 狠狠v欧美ⅴ日韩v亚洲v大胸| 亚洲精品免费视频| 免费国产成人看片在线| 日本三级一区| 欧美午夜片欧美片在线观看| 亚洲午夜无码av毛片久久| 天天看天天摸天天操| 欧美日韩调教| 色综合久久久久久中文网| 日韩**一区毛片| 精品欧美一区二区久久| 欧美老熟妇乱大交xxxxx| eeuss国产一区二区三区四区| 欧美精品三级在线观看| 污污内射在线观看一区二区少妇| 久久亚洲国产| 欧美做受高潮电影o| 午夜免费福利视频| 丁香桃色午夜亚洲一区二区三区| 粉嫩高清一区二区三区精品视频| 精品区在线观看| 国产精品嫩草影院av蜜臀| 亚洲精品一区二区毛豆| 黄色网址在线免费播放| 亚洲精品欧美综合四区| 天天干在线影院| 亚洲盗摄视频| 欧美在线精品免播放器视频| 亚洲精品综合网| 亚洲精品福利视频网站| 欧美女同在线观看| 成人97精品毛片免费看| 日韩精品一区二区在线观看| 在线xxxxx| 91成人网在线观看| 成人国产精品久久久| av中文字幕一区二区三区| 日韩欧美在线播放| 黄色录像a级片| 国产欧美大片| 国产精品777| 先锋av资源站| 欧美国产日本视频| 欧美美女黄色网| 国产中文在线播放| 亚洲第一av在线| 娇妻被老王脔到高潮失禁视频| 国产模特精品视频久久久久| 国产三级精品在线不卡| 国产免费拔擦拔擦8x在线播放| 亚洲成人a级网| 亚洲免费激情视频| 另类调教123区| 国产日韩欧美一区二区三区四区 | 欧美成人黄色小视频| 国产露脸无套对白在线播放| 成人av在线影院| 亚洲人精品午夜射精日韩 | 成人精品视频久久久久| 日韩伦理在线观看| 日韩一区国产二区欧美三区| 青娱乐免费在线视频| 香蕉国产精品偷在线观看不卡| 国产精品吴梦梦| 嫩草香蕉在线91一二三区| 舔着乳尖日韩一区| 中文字幕人妻一区二区| 日韩国产成人精品| 一区二区精品国产| 国产高清不卡| 中文字幕亚洲专区| 国产wwwwwww| 红桃视频成人在线观看| 精品成人av一区二区三区| 免费高清成人在线| 亚洲乱码日产精品bd在线观看| 欧美三级自拍| 国产色视频一区| av在线私库| 亚洲丝袜av一区| 久久黄色精品视频| 高清在线成人网| 视色,视色影院,视色影库,视色网| 日韩精品亚洲专区在线观看| 欧美壮男野外gaytube| 日本成人网址| 日韩精品免费视频| 国产对白videos麻豆高潮| 国产一区二区按摩在线观看| 男女啪啪免费视频网站| 成人av动漫在线观看| 1卡2卡3卡精品视频| 性国裸体高清亚洲| 久久成人人人人精品欧| 亚洲一卡二卡在线观看| 亚洲午夜av在线| 亚洲精品成人无码毛片| 91精品一区国产高清在线gif| 国产综合精品一区二区三区| 国产亚洲欧美日韩精品一区二区三区 | 亚洲人精品午夜| 亚洲专区区免费| 国产二区国产一区在线观看| 天天成人综合网| 天天综合在线观看| 78色国产精品| 欧美日本网站| 欧美在线|欧美| 色噜噜噜噜噜噜| 99视频超级精品| 日韩a在线播放| 欧美日本不卡高清| 亚洲一区二区不卡视频| 国产精品一区二区av交换| 国产精品久久久91| 求av网址在线观看| 亚洲美女久久久| 蜜桃视频污在线观看| 欧美精品aⅴ在线视频| 波多野结衣网站| 国产精品女主播在线观看| 色婷婷免费视频| 成人免费视频国产在线观看| 日韩av影视大全| 国产一区日韩欧美| 天天综合中文字幕| 粉嫩一区二区三区在线观看| 国产精品美女久久久久久免费| 婷婷成人激情| 在线电影欧美日韩一区二区私密| 国产乱淫av片免费| 欧美日韩在线不卡| 91在线看视频| 婷婷六月综合亚洲| 亚洲欧美在线观看视频| 亚洲一区二区三区爽爽爽爽爽 | 国产欧美最新羞羞视频在线观看| 欧美黑人粗大| 一区二区三区视频免费| 日本亚洲一区| 91麻豆精品国产91久久久久久久久 | 精品人妻一区二区免费视频| 高清视频一区二区| 在线xxxxx| 99国产精品久| 欧美一级xxxx| aa级大片欧美三级| 99热自拍偷拍| 91精品国产自产在线观看永久∴| 中国成人亚色综合网站| 久久久五月天| 日本天堂免费a| 欧美日韩亚洲一区三区| 毛片在线播放视频| 久久久天天操| 国产精品igao激情视频| 亚洲午夜在线| 国产亚洲欧美在线视频| 日韩在线一区二区| 91高清国产视频| 国产精品一区一区| 91日韩视频在线观看| 麻豆精品视频在线观看| 国内精品国产三级国产aⅴ久| 丁香另类激情小说| 疯狂揉花蒂控制高潮h| 国内精品写真在线观看| 日韩欧美xxxx| 国产日韩一区| 日本999视频| 香蕉精品999视频一区二区| 丰满少妇在线观看| 国产美女在线精品| 亚洲调教欧美在线| 国产精品国产三级国产普通话99| 欧美又粗又大又长| 亚洲欧美综合在线精品| 久久久全国免费视频| 日韩欧美999| 国产精品主播一区二区| 日韩av在线影院| 日本a在线播放| 91wwwcom在线观看| 99视频这里有精品| 国产日韩欧美成人| 国产精品x8x8一区二区| 成人女人免费毛片| 国产探花一区在线观看| 国产精品久久成人免费观看| 亚洲欧美日韩国产一区二区| 在线观看日本www| 久久综合色婷婷| 国产精品九九九九九九| 欧美午夜精品久久久久久孕妇| 免费av网站观看| 日日狠狠久久偷偷四色综合免费| av影片免费在线观看| 一本色道久久88精品综合| 最爽无遮挡行房视频在线| 欧美成人免费一级人片100| 97se综合| 国产精品一区二区欧美| 欧美国产一级| 欧美黄网站在线观看| 成人手机电影网| 天天鲁一鲁摸一摸爽一爽| 日韩欧美中文免费| 亚洲欧美另类视频| 亚洲国产精品99久久| 久热国产在线| 国产精品久久久久999| 欧美a级网站| 97视频在线免费| 久久经典综合| 国产又黄又猛又粗| 久久综合色8888| 好吊操这里只有精品| 精品人在线二区三区| 18在线观看的| 亚洲精品欧美日韩专区| 国产精品极品国产中出| 亚洲天堂av免费在线观看| 日本成人超碰在线观看| 亚洲天堂视频一区| 精品福利一区二区| 人妻少妇一区二区三区| 欧美美女15p| 97视频一区| 成人免费在线视频播放| 久久高清免费观看| 给我看免费高清在线观看| 亚洲r级在线视频| 老熟妇一区二区三区啪啪| 亚洲精品一区中文| 免费观看成人高潮| 国产免费亚洲高清| 999精品色在线播放| 精品久久久99| 亚洲色图欧洲色图婷婷| 国产乱淫a∨片免费视频| 欧美精品一区在线播放| 99re热精品视频| av高清在线免费观看| 成人一区二区在线观看| 成年人免费看毛片| 日韩av在线精品| 浪潮色综合久久天堂| 国产精品免费观看高清| 亚洲精品美女| www.污网站| 亚洲靠逼com| 亚洲卡一卡二卡三| 欧美一级高清免费播放| 精品国产一区二区三区2021| 麻豆精品视频| 久久精品首页| 人妻无码一区二区三区免费| 欧美精品日日鲁夜夜添| 人妖欧美1区| 久久综合狠狠综合久久综青草| 视频在线观看国产精品| 2014亚洲天堂| 色av综合在线| 波多野结衣在线网站| 91丝袜美腿美女视频网站| 欧美猛男男男激情videos| 日本大片免费看| 97国产一区二区| 中文字幕有码视频| 精品视频偷偷看在线观看| 日韩欧美一区二区三区免费观看| 正在播放精油久久| 99re66热这里只有精品3直播| 中文在线最新版天堂| 久久国产精品影视| 网友自拍一区| 天美一区二区三区| 欧美小视频在线| av免费在线网站| 欧美精品久久| 国产精品一二三区在线| 亚洲欧美一区二区三区在线观看| 精品国产欧美成人夜夜嗨| 久久悠悠精品综合网| 四季av一区二区三区| 精品久久久久人成| 国产在线高清理伦片a| 欧美高清性xxxxhd| 国产iv一区二区三区| 国产精品成人无码|