查看: 2356|回复: 0

[ASP.NET教程] 从源码浅析MVC的MvcRouteHandler、MvcHandler和MvcHttpHandler

发表于 2017-1-11 19:14:34

熟悉WebForm开发的朋友一定都知道,Page类必须实现一个接口,就是IHttpHandler。HttpHandler是一个HTTP请求的真正处理中心,在HttpHandler容器中,asp.net Framework调用HttpHandler的ProcessRequest成员方法来对这个HTTP请求进行真正的处理,真正地对客户端请求的服务器页面做出编译和执行。归根结底,Asp.Net所有的HTTP请求最后都是由IHttpHandler的实现来处理的,Asp.Net MVC框架当然也不例外。下面就从MvcRouteHandler、MvcHandler和MvcHttpHandler三个常见的Handler类来浅析一下MVC是怎么和IHttpHandler联系起来进行HTTP请求处理的。

1、MvcRouteHandler

在MVC中MvcApplication通常在Application_Start事件里实现RegisterRoutes方法:

  1. protected void Application_Start()
  2. {
  3. AreaRegistration.RegisterAllAreas();
  4. RegisterGlobalFilters(GlobalFilters.Filters);
  5. RegisterRoutes(RouteTable.Routes);
  6. }
复制代码

其中RegisterRoutes方法通常类似如下:

  1. public static void RegisterRoutes(RouteCollection routes)
  2. {
  3. routes.IgnoreRoute("{resource}.axd/{*pathInfo}");
  4. routes.MapRoute(
  5. "Default", // Route name
  6. "{controller}/{action}/{id}", // URL with parameters
  7. new { controller = "Home", action = "Index", id = UrlParameter.Optional } // Parameter defaults
  8. );
  9. }
复制代码

这个方法非常重要,因为正是从这里开始,Asp.Net应用程序可以和Asp.Net URL Routing组件完美结合起来实现Url优化(MVC和WebForm都可以,可以参考dudu的这一篇)。

从MVC源码入手,我们发现RouteCollection的MapRoute扩展方法最终都是通过RouteCollectionExtensions的一个静态方法实现的:

  1. [SuppressMessage("Microsoft.Design", "CA1054:UriParametersShouldNotBeStrings", MessageId = "2#", Justification = "This is not a regular URL as it may contain special routing characters.")]
  2. public static Route MapRoute(this RouteCollection routes, string name, string url, object defaults, object constraints, string[] namespaces)
  3. {
  4. if (routes == null)
  5. {
  6. throw new ArgumentNullException("routes");
  7. }
  8. if (url == null)
  9. {
  10. throw new ArgumentNullException("url");
  11. }
  12. Route route = new Route(url, new MvcRouteHandler())
  13. {
  14. Defaults = new RouteValueDictionary(defaults),
  15. Constraints = new RouteValueDictionary(constraints),
  16. DataTokens = new RouteValueDictionary()
  17. };
  18. if ((namespaces != null) && (namespaces.Length > 0))
  19. {
  20. route.DataTokens["Namespaces"] = namespaces;
  21. }
  22. routes.Add(name, route);
  23. return route;
  24. }
复制代码

大家注意这一行: Route route = new Route(url, new MvcRouteHandler()),正是通过MvcRouteHandler(其实从命名就可以猜到),Asp.Net的URL Routing组件就和IHttpHandler有了关联。到这里我们猜测,MvcRouteHandler一定和IHttpHandler有关系(甚至它可能就是IHttpHandler的一个具体实现),而且URL Routing组件最后一定映射到一个IHttpHandler处理程序来处理相应的HTTP请求。我们来查看MvcRouteHandler具体源码实现印证一下自己的看法:

  1. /* ****************************************************************************
  2. *
  3. * Copyright (c) Microsoft Corporation. All rights reserved.
  4. *
  5. * This software is subject to the Microsoft Public License (Ms-PL).
  6. * A copy of the license can be found in the license.htm file included
  7. * in this distribution.
  8. *
  9. * You must not remove this notice, or any other, from this software.
  10. *
  11. * ***************************************************************************/
  12. namespace System.Web.Mvc
  13. {
  14. using System.Web.Routing;
  15. using System.Web.SessionState;
  16. public class MvcRouteHandler : IRouteHandler
  17. {
  18. private IControllerFactory _controllerFactory;
  19. public MvcRouteHandler()
  20. {
  21. }
  22. public MvcRouteHandler(IControllerFactory controllerFactory)
  23. {
  24. _controllerFactory = controllerFactory;
  25. }
  26. protected virtual IHttpHandler GetHttpHandler(RequestContext requestContext)
  27. {
  28. requestContext.HttpContext.SetSessionStateBehavior(GetSessionStateBehavior(requestContext));
  29. return new MvcHandler(requestContext);
  30. }
  31. protected virtual SessionStateBehavior GetSessionStateBehavior(RequestContext requestContext)
  32. {
  33. string controllerName = (string)requestContext.RouteData.Values["controller"];
  34. IControllerFactory controllerFactory = _controllerFactory ?? ControllerBuilder.Current.GetControllerFactory();
  35. return controllerFactory.GetControllerSessionBehavior(requestContext, controllerName);
  36. }
  37. #region IRouteHandler Members
  38. IHttpHandler IRouteHandler.GetHttpHandler(RequestContext requestContext)
  39. {
  40. return GetHttpHandler(requestContext);
  41. }
  42. #endregion
  43. }
  44. }
复制代码

可惜,MvcRouteHandler没有继承实现IHttpHandler接口,而是继承了IRouteHandler接口,但是我们欣喜发现IRouteHandler接口的唯一方法GetHttpHandler返回了一个IHttpHandler,这实在是太让人感到意料之中的一阵激动了。

2、MvcHandler

继续上面MvcRouteHandler的源码,我们发现MvcRouteHandler具体实现GetHttpHandler的时候最后new了一个MvcHandler对象返回:

  1. protected virtual IHttpHandler GetHttpHandler(RequestContext requestContext)
  2. {
  3. requestContext.HttpContext.SetSessionStateBehavior(GetSessionStateBehavior(requestContext));
  4. return new MvcHandler(requestContext);
  5. }
复制代码

也就是说MvcHandler是IHttpHandler子类确定无疑了。然后查看MvcHandler源码,发现MvcHandler继承实现了IHttpAsyncHandler, IHttpHandler, IRequiresSessionState三个无比熟悉的接口。而这三个接口如果都实现了,在MVC框架下是不是任何http请求就可以通吃了呢?从MSDN我们得知,事实不是这样的:

注意,即使 MvcHandler 实现 IHttpHandler,也不能将其映射为处理程序(例如.mvc 文件扩展名),因为该类不支持无参数构造函数。 (它唯一的构造函数需要一个 RequestContext 对象)

但是,还好,我们还有MvcHttpHandler。

3、MvcHttpHandler

如你所知,MvcHttpHandler可以“弥补”MvcHandler的不足,为什么这样说呢?其实2中也提到过了,MvcHandler没有无参的构造函数,因此即使MvcHandler实现了 IHttpHandler接口,在IIS中也不能将其映射为某类文件扩展名的处理程序,需要结合路由模块使用。

而MvcHttpHandler就提供了不通过路由模块的情况下直接处理映射的处理程序。通过查看MvcHttpHandler源码我们发现,MvcHttpHandler继承实现了UrlRoutingHandler, IHttpAsyncHandler, IRequiresSessionState接口,而UrlRoutingHandler继承自IHttpHandler。MvcHttpHandler带无参的构造函数(也就是说我们可以直接new一个MvcHttpHandler无参数对象?不用担心上下文?希望这里的解释不是令人感到费解),而且继承自UrlRoutingHandler类实现了IHttpHandler接口,因此可以在ASP.NET程序中让你更加灵活使用用来解决一些问题。

参考:

http://msdn.microsoft.com/zh-cn/library/system.web.mvc.mvcroutehandler.aspx

http://msdn.microsoft.com/zh-cn/library/system.web.mvc.mvchandler.aspx

http://msdn.microsoft.com/zh-cn/library/system.web.mvc.mvchttphandler.aspx



回复

使用道具 举报