一、前言

  对于WebForm开发,请求通常是一个以.aspx结尾的url,对应一个物理文件,从代码的角度来说它其实是一个控件(Page)。而在MVC中,一个请求对应的是一个Controller里的Action。熟悉asp.net的朋友都知道,asp.net请求实际都是交给HttpHandler处理(实现了IHttpHandler的类型)。无论是.aspx,.ashx,.asmx 还是MVC里的Action,请求都会交给HttpHandler。具体是在管道事件中,会根据请求创建一个HttpHandler,并执行它的PR方法。对于aspx和ashx都很好理解,因为它们本身就实现了IHttpHandler接口,而MVC的Controller和Action都和HttpHandler没有关系,它是如何实现的呢?接下来我们就看一个请求是如何进入mvc框架内部的。

二、例子

  WebForm和MVC都是建立在asp.net平台上的,Webform出现得比较早,那么MVC是如何做到在不影响底层框架,实现扩展的呢?这主要得益于asp.net的路由机制。路由机制并不属于MVC,WebForm也可以使用它。它的目的是让一个请求与物理文件分离,原理是通过映射关系,将请求映射到指定的HttpHandler。例如我们也可以将一个/Admin/User.aspx?name=张三 的请求映射成可读性更好的/Admin/张三。下面是两种url的注册方式:

  1. public static void RegisterRoutes(RouteCollection routes)
  2. {
  3. //MVC
  4. routes.MapRoute(
  5. name: "Default",
  6. url: "{controller}/{action}/{id}",
  7. defaults: new { controller = "Home", action = "Index", id = UrlParameter.Optional }
  8. );
  9.  
  10. //WebForm
  11. routes.MapPageRoute(
  12. routeName: "WebForm",
  13. routeUrl: "Admin/{user}",
  14. physicalFile: "~/Admin/User.aspx"
  15. );
  16. }

  RouteCollection是一个Route集合,Route封装了名称、url模式、约束条件、默认值等路由相关信息。其中,MapPageRoute是RouteCollection定义的方法,而MapRoute是MVC扩展出来的(扩展方法的好处就是可以在不修改原有代码的情况下添加所需的功能)。它们的目的都是一样的,创建一个Route对象,添加到集合当中;我们也可以new 一个Route对象,然后调用RouteCollection.Add,效果是一样的。下面我们主要关注MVC的实现过程,WebForm其实也是类似的。

三、分析源码

  接下来我们看MVC是如何利用路由机制实现扩展的。路由机制是通过一个UrlRoutingModule完成的,它是一个实现了IHttpModule的类,路由模块已经默认帮我们注册好了。HttpModule通过注册HttpApplication事件参与到管道处理请求中,具体是订阅HttpApplication某个阶段的事件。路由机制就是利用这个原理,UrlRoutingModule订阅了PostResolveRequestCache 事件,实现url的映射。为什么是该事件呢?因为该事件的下一步就要完成请求和物理文件的映射,所以必须要此之前进行拦截。核心代码如下:

  1. public class UrlRoutingModule : IHttpModule {
  2. public RouteCollection RouteCollection {
  3. get {
  4. if (_routeCollection == null) {
  5. //全局的RouteCollection集合
  6. _routeCollection = RouteTable.Routes;
  7. }
  8. return _routeCollection;
  9. }
  10. set {
  11. _routeCollection = value;
  12. }
  13. }
  14.  
  15. protected virtual void Init(HttpApplication application) {
  16. //注册PostResolveRequestCache事件
  17. application.PostResolveRequestCache += OnApplicationPostResolveRequestCache;
  18. }
  19.  
  20. private void OnApplicationPostResolveRequestCache(object sender, EventArgs e) {
  21. //创建上下文
  22. HttpApplication app = (HttpApplication)sender;
  23. HttpContextBase context = new HttpContextWrapper(app.Context);
  24. PostResolveRequestCache(context);
  25. }
  26.  
  27. public virtual void PostResolveRequestCache(HttpContextBase context) {
  28. //1.获取RouteData
  29. RouteData routeData = RouteCollection.GetRouteData(context);
  30. if (routeData == null) {
  31. return;
  32. }
  33. //2.获取IRouteHandler
  34. IRouteHandler routeHandler = routeData.RouteHandler;
  35. if (routeHandler == null) {
  36.  
  37. }
  38.  
  39. //RequestContext保证了HttpContext和RouteData,在后续使用
  40. RequestContext requestContext = new RequestContext(context, routeData);
  41.  
  42. context.Request.RequestContext = requestContext;
  43.  
  44. //3.获取IHttpHandler
  45. IHttpHandler httpHandler = routeHandler.GetHttpHandler(requestContext);
  46.  
  47. //重新映射到处理程序
  48. context.RemapHandler(httpHandler);
  49. }
  50. }  

  我们关注主要方法PostResolveRequestCache,这里有三个关键步骤。

  步骤一. 获取RouteData

  RouteData是对Route的包装,在后续的处理中使用。它的获取是通过RouteCollection获得的,这个和上面注册用到的RouteTable.Routes是同一个集合对象。调用RouteCollection的GetRouteData会遍历它的每一个项,也就是Route对象,然后调用Route对象的GetRouteData方法(MVC内部很多集合都用到了这种设计)。如下代码:

  1. public RouteData GetRouteData(HttpContextBase httpContext) {
  2. using (GetReadLock()) {
  3. foreach (RouteBase route in this) {
  4. RouteData routeData = route.GetRouteData(httpContext);
  5. if (routeData != null) {
  6. return routeData;
  7. }
  8. }
  9. }
  10. return null;
  11. }

  Route对象的GetRouteData方法如下:

  1. public override RouteData GetRouteData(HttpContextBase httpContext) {
  2. string requestPath = httpContext.Request.AppRelativeCurrentExecutionFilePath.Substring(2) + httpContext.Request.PathInfo;
  3.  
  4. //结合默认值,匹配url
  5. RouteValueDictionary values = _parsedRoute.Match(requestPath, Defaults);
  6.  
  7. if (values == null) {
  8. return null;
  9. }
  10.  
  11. //包装成RouteData,这里为什么不放在if后面呢?
  12. RouteData routeData = new RouteData(this, RouteHandler);
  13.  
  14. //匹配约束
  15. if (!ProcessConstraints(httpContext, values, RouteDirection.IncomingRequest)) {
  16. return null;
  17. }
  18.  
  19. //RouteData的Values和DataTokens都来自于Route
  20. foreach (var value in values) {
  21. routeData.Values.Add(value.Key, value.Value);
  22. }
  23. if (DataTokens != null) {
  24. foreach (var prop in DataTokens) {
  25. routeData.DataTokens[prop.Key] = prop.Value;
  26. }
  27. }
  28.  
  29. return routeData;
  30. }

  可以看到,Route对象的GetRouteData方法会匹配url模式,和检查约束条件,如何不符合会返回null。如果匹配,则new一个RouteData。

  步骤二、获取IRouteHandler接口对象

  上面创建RouteData,参数分别是当前Route对象和它的RouteHandler属性。RouteHandler是一个IRouteHandler,这是一个重要接口,它的定义如下:

  1. public interface IRouteHandler {
  2. IHttpHandler GetHttpHandler(RequestContext requestContext);
  3. }

  很明显,它是用于获取IHttpHandler的。那么Route对象的RouteHandler属性又是在哪里初始化的呢?我们回到开始的注册方法,routes.MapRoute,这个方法根据传递的参数创建一个Route对象,该方法的实现如下:

  1. public static Route MapRoute(this RouteCollection routes, string name, string url, object defaults, object constraints, string[] namespaces)
  2. {
  3. //创建一个Route对象,它的IRouteHandler为MvcRouteHandler
  4. Route route = new Route(url, new MvcRouteHandler())
  5. {
  6. Defaults = CreateRouteValueDictionary(defaults),
  7. Constraints = CreateRouteValueDictionary(constraints),
  8. DataTokens = new RouteValueDictionary()
  9. };
  10.  
  11. if ((namespaces != null) && (namespaces.Length > 0))
  12. {
  13. route.DataTokens["Namespaces"] = namespaces;
  14. }
  15.  
  16. //将Route注册到RouteCollection中
  17. routes.Add(name, route);
  18.  
  19. return route;
  20. }

  在创建Route时,除了传递url模式外,还默认帮我们传递了一个MvcRouteHandler,它实现了IRouteHandler接口。
  步骤三、获取IHttpHandler接口对象

  有了MvcRouteHandler,就可以调用它的GetHttpHandler方法获取IHttpHandler了,该方法实现如下:

  1. protected virtual IHttpHandler GetHttpHandler(RequestContext requestContext)
  2. {
  3. //设置session状态
  4. requestContext.HttpContext.SetSessionStateBehavior(GetSessionStateBehavior(requestContext));
  5.  
  6. //返回一个实现了IHttpHandler的MvcHandler
  7. return new MvcHandler(requestContext);
  8. }

  可以看到,它返回了一个MvcHandler,MvcHandler就实现了IHttpHandler接口。所以开头说的,请求本质都是交给HttpHandler的,其实MVC也是这样的,请求交给了MvcHandler处理。我们可以看MvcHandler定义和主要方法:

  1. public class MvcHandler : IHttpAsyncHandler, IHttpHandler, IRequiresSessionState
  2. {
  3. protected internal virtual IAsyncResult BeginProcessRequest(HttpContextBase httpContext, AsyncCallback callback, object state)
  4. {
  5. IController controller;
  6. IControllerFactory factory;
  7.  
  8. //这个方法里会激活Controller对象
  9. ProcessRequestInit(httpContext, out controller, out factory);
  10.  
  11. IAsyncController asyncController = controller as IAsyncController;
  12. if (asyncController != null)
  13. {
  14. // asynchronous controller
  15. BeginInvokeDelegate beginDelegate = delegate(AsyncCallback asyncCallback, object asyncState)
  16. {
  17. try
  18. {
  19. //调用Controller的BeginExecute方法
  20. return asyncController.BeginExecute(RequestContext, asyncCallback, asyncState);
  21. }
  22. catch
  23. {
  24. factory.ReleaseController(asyncController);
  25. throw;
  26. }
  27. };
  28.  
  29. EndInvokeDelegate endDelegate = delegate(IAsyncResult asyncResult)
  30. {
  31. try
  32. {
  33. asyncController.EndExecute(asyncResult);
  34. }
  35. finally
  36. {
  37. factory.ReleaseController(asyncController);
  38. }
  39. };
  40.  
  41. SynchronizationContext syncContext = SynchronizationContextUtil.GetSynchronizationContext();
  42. AsyncCallback newCallback = AsyncUtil.WrapCallbackForSynchronizedExecution(callback, syncContext);
  43. return AsyncResultWrapper.Begin(newCallback, state, beginDelegate, endDelegate, _processRequestTag);
  44. }
  45. else
  46. {
  47. // synchronous controller
  48. Action action = delegate
  49. {
  50. try
  51. {
  52. controller.Execute(RequestContext);
  53. }
  54. finally
  55. {
  56. factory.ReleaseController(controller);
  57. }
  58. };
  59.  
  60. return AsyncResultWrapper.BeginSynchronous(callback, state, action, _processRequestTag);
  61. }
  62. }
  63. }

  可以看到,MvcHandler的任务就是激活Controller,并执行它的Execute方法。这个过程和Webform里的页面处理是很相似的,.aspx请求到来,会根据虚拟路径找到实现IHttpHandler的Page(类似于路由机制根据url模式找到MvcHandler),然后进入Page的页面周期(类似于Mvc的激活Controller,然后执行Action过程)。

四、总结

接下来,简单总结一下请求进入到MVC框架的过程:

1.添加路由对象Route到全局的RouteCollection,Route的IRouteHandler初始化为MvcRouteHandler。

2. UrlRoutingModule注册 HttpApplication PostResolveRequestCache事件,实现请求拦截。

3. 请求到来, 在处理事件中遍历RouteCollection,调用每一个Route对象的GetRouteData获取RouteData包装对象。

4. 调用MvcRouteHandler的GetHttpHandler获取MvcHandler。

5. 调用HttpContext的RemapHandler将请求映射到MvcHandler处理程序。

6. 执行MvcHandler的PR方法,激活Controller,执行Action。

请求如何进入ASP.NET MVC框架的更多相关文章

  1. 在ASP.NET MVC 框架中调用 html文件及解析get请求中的参数值

    在ASP.NET MVC 框架中调用 html文件: public ActionResult Index() { using (StreamReader sr = new StreamReader(P ...

  2. 写自己的ASP.NET MVC框架(上)

    http://www.cnblogs.com/fish-li/archive/2012/02/12/2348395.html 阅读目录 开始 ASP.NET程序的几种开发方式 介绍我的MVC框架 我的 ...

  3. 学习“迷你ASP.NET MVC框架”后的小结

    看蒋老师MVC的书第二个大收获可以是算是看了这个迷你ASP.NET MVC框架了,虽然它远不如真正ASP.NET MVC(下文简称“MVC”)那么复杂庞大,但在迷你版中绕来绕去也够呛的.这部分我看了几 ...

  4. BrnShop开源网上商城第二讲:ASP.NET MVC框架

    在团队设计BrnShop的web项目之初,我们碰到了两个问题,第一个是数据的复用和传递,第二个是大mvc框架和小mvc框架的选择.下面我依次来说明下. 首先是数据的复用和传递:对于BrnShop的每一 ...

  5. 写自己的ASP.NET MVC框架(下)

    上篇博客[写自己的ASP.NET MVC框架(上)] 我给大家介绍我的MVC框架对于Ajax的支持与实现原理.今天的博客将介绍我的MVC框架对UI部分的支持. 注意:由于这篇博客是基于前篇博客的,因此 ...

  6. 如何用asp.net MVC框架、highChart库从sql server数据库获取数据动态生成柱状图

    如何用asp.net MVC框架.highChart库从sql server数据库获取数据动态生成柱状图?效果大概是这样的,如图: 请问大侠这个这么实现呢?

  7. ASP.NET MVC框架开发系列课程 (webcast视频下载)

    课程讲师: 赵劼 MSDN特邀讲师 赵劼(网名“老赵”.英文名“Jeffrey Zhao”,技术博客为http://jeffreyzhao.cnblogs.com),微软最有价值专家(ASP.NET ...

  8. 【转】ASP.NET MVC框架下使用MVVM模式-KnockOutJS+JQ模板例子

    KnockOutJS学习系列----(一) 好几个月没去写博客了,最近也是因为项目紧张,不过这个不是借口,J. 很多时候可能是因为事情一多,然后没法静下来心来去写点东西,学点东西. 也很抱歉,突然看到 ...

  9. 【工作笔记二】ASP.NET MVC框架下使用MVVM模式

    ASP.NET MVC框架下使用MVVM模式 原文:http://www.cnblogs.com/n-pei/archive/2011/07/21/2113022.html 对于asp.net mvc ...

随机推荐

  1. Federated Identity Pattern 联合身份模式

    Delegate authentication to an external identity provider. This pattern can simplify development, min ...

  2. SSM项目搭建(提供源码)

    1创建web动态项目,项目结构截图 2.配置日志文件 #\u5B9A\u4E49LOG\u8F93\u51FA\u7EA7\u522B log4j.rootLogger=INFO,Console,Fi ...

  3. Java进击C#——应用开发之Asp.net MVC

    本章简言 上一章笔者讲到关于Asp.NET的知识点.了解Asp.NET基本的知识点之后,我们在来学习关于C#的MVC框架就简单多了.显然本章就是来介绍一下关于Asp.NET MVC.对于MVC的思想笔 ...

  4. 数百个 HTML5 例子学习 HT 图形组件 – WebGL 3D 篇

    <数百个 HTML5 例子学习 HT 图形组件 – 拓扑图篇>一文让读者了解了 HT的 2D 拓扑图组件使用,本文将对 HT 的 3D 功能做个综合性的介绍,以便初学者可快速上手使用 HT ...

  5. IDEA上安装和使用checkstyle,findbugs,visualVM,PMD插件

    ##安装插件步骤: 1.打开settings 2.选择plugins 3.点击"Browse repositories" 4.搜索对应内插件,点击"install&quo ...

  6. FastDFS 安装及使用

    FastDFS 安装及使用 2012-11-17 13:10:31|  分类: Linux|举报|字号 订阅     Google了一下,流行的开源分布式文件系统有很多,介绍如下:   mogileF ...

  7. 开始我的IT博客之旅

    这是一个好的开始,过程很漫长,但我却乐在其中. 在大学之际,这是我的又一个开始,随便写点啦. 想把每一次的过程记录下来 这样以后对自己 对别人都会有所帮助. 好啦 作为一名大三的学生党  加油吧!

  8. Canvas的width,height 和 样式中Canvas的width,height

    控制Canvas的大小,有两种方式: 1:直接设置Canvas标签上的书width,height属性值; 2:通过Css设置Canvas的width,height; 这两种方式,区别是很大的. 1:C ...

  9. CSS布局 -- 圣杯布局 & 双飞翼布局

    按照我的理解,其实圣杯布局跟双飞翼布局的实现,目的都是左右两栏固定宽度,中间部分自适应. 但在这里实现起来还是有一些区别的 [圣杯布局] 在这里,实现了左(200px) 右(220px) 宽度固定,中 ...

  10. UITabBarController 升级定制

    UITabBarController 定制 特点 用法 1.准备工作: 加入你的相关图片,放入了Assets.xcassets; 导入Categroy文件夹(这个里面的文件,在这里不详细说明了,有疑问 ...