URL路由系统通过对请求地址进行解析从而得到以目标Controller名称为核心的路由数据。Url路由系统最初是为了实现请求url与物理文件路径分离而建立的,MVC的Url Route是将Url地址与物理文件映射转移到了目标Controller的映射。

Url路由不是ASP.NET MVC特有的,而是建立在ASP.NET上面的,MVC的只是对这个路由的拓展使用(asp.net也开始使用这拓展了)。

  我们在App_Start文件夹中找到RouteConfig.cs的文件,打开看

 1         public static void RegisterRoutes(RouteCollection routes)
 2         {
 3             routes.IgnoreRoute("{resource}.axd/{*pathInfo}");
 4
 5             routes.MapRoute(
 6                 name: "Default",
 7                 url: "{controller}/{action}/{id}",
 8                 defaults: new { controller = "Home", action = "Index", id = UrlParameter.Optional }
 9             );
10         }

  大体可以猜出什么意思,url那一栏地址中第一个是controller,第二个是action,第三个是参数id,defaults是默认的参数,然后在Global.asax对该路由进行注册。

  

1         protected void Application_Start()
2         {
3             AreaRegistration.RegisterAllAreas();
4             FilterConfig.RegisterGlobalFilters(GlobalFilters.Filters);
5             RouteConfig.RegisterRoutes(RouteTable.Routes);
6             BundleConfig.RegisterBundles(BundleTable.Bundles);
7         }

  

  简单来说路由的任务:检查请求的URL,找出当前请求的是哪个controller中的哪个action,并且有无带了什么参数过来。

  那好,我们来分析一下请求一个地址的时候,路由系统发生了什么事情?我们先来大胆的猜一下:类似于网络设备的路由器,它会有一个路由表,根据我们配置的规则对端口过来的数据进行转发,这张表里就记录了匹配规则跟处理的程序,当一个数据包过来的时候,去这张表里面寻找所相应的发送地址,找到的话路由系统就将这数据包发往哪个对应的地址里面。

  按照上面逻辑的话,我们先来找找匹配的规则在哪里加上来的,我们在上面的静态方法里面看到有一个RouteCollection的东西,进去看看里面有啥?

  

  发现了一个MapPageRoute的方法

  

 1         //
 2         // 摘要:
 3         //     提供用于定义 Web 窗体应用程序的路由的方法。
 4         //
 5         // 参数:
 6         //   routeName:
 7         //     路由的名称。
 8         //
 9         //   routeUrl:
10         //     路由的 URL 模式。
11         //
12         //   physicalFile:
13         //     路由的物理 URL。
14         //
15         // 返回结果:
16         //     将添加到路由集合的路由。
17         public Route MapPageRoute(string routeName, string routeUrl, string physicalFile);

  

  可以看到我们调用RouteCollection的MapPageRoute方法将某个物理文件路径映射到一个URL模板上,这个过程其实就是基于指定的url模板创建一个路由对象,并且将他添加到这个全局路由表中,那程序是在哪里有个Add的入口呢?不小心在RouteCollection里面发现了一个Add的方法

  

 1         // 摘要:
 2         //     将路由添加到 System.Web.Routing.RouteCollection 对象的结尾,并为路由分配指定的名称。
 3         //
 4         // 参数:
 5         //   name:
 6         //     标识路由的值。 该值可为 null 或空字符串。
 7         //
 8         //   item:
 9         //     要添加到集合结尾的路由。
10         //
11         // 异常:
12         //   System.ArgumentNullException:
13         //     item 为 null。
14         //
15         //   System.ArgumentException:
16         //     name 已在集合中使用。
17         public void Add(string name, RouteBase item);

  

  可以看到这里添加了一个Routebase的东西,我们点击去看看是什么来头?

  

 1     // 摘要:
 2     //     用作表示 ASP.NET 路由的所有类的基类。
 3     [TypeForwardedFrom("System.Web.Routing, Version=3.5.0.0, Culture=Neutral, PublicKeyToken=31bf3856ad364e35")]
 4     public abstract class RouteBase
 5     {
 6         // 摘要:
 7         //     初始化该类供继承的类实例使用。 此构造函数只能由继承的类调用。
 8         protected RouteBase();
 9
10         // 摘要:
11         //     获取或设置一个值,该值指示 ASP.NET 路由操作是否应处理与现有文件匹配的 URL。
12         //
13         // 返回结果:
14         //     如果 ASP.NET 路由操作处理所有请求(甚至包括与现有文件匹配的请求),则为 true;否则为 false。 默认值为 false。
15         public bool RouteExistingFiles { get; set; }
16
17         // 摘要:
18         //     当在派生类中重写时,会返回有关请求的路由信息。
19         //
20         // 参数:
21         //   httpContext:
22         //     一个对象,封装有关 HTTP 请求的信息。
23         //
24         // 返回结果:
25         //     一个对象,包含路由定义的值(如果该路由与当前请求匹配)或 null(如果该路由与请求不匹配)。
26         public abstract RouteData GetRouteData(HttpContextBase httpContext);
27         //
28         // 摘要:
29         //     当在派生类中重写时,会检查路由是否与指定值匹配,如果匹配,则生成一个 URL,然后检索有关该路由的信息。
30         //
31         // 参数:
32         //   requestContext:
33         //     一个对象,封装有关所请求的路由的信息。
34         //
35         //   values:
36         //     一个包含路由参数的对象。
37         //
38         // 返回结果:
39         //     一个对象(包含生成的 URL 和有关路由的信息)或 null(如果路由与 values 不匹配)。
40         public abstract VirtualPathData GetVirtualPath(RequestContext requestContext, RouteValueDictionary values);
41     }

  我们看到了两个抽象方法:GetRouteData与GetVirtualPath,当一个请求过来的时候,它们都会去根据URL模板的模式与代表请求地址的URL地址进行匹配,如果匹配失败返回null,成功的话,GetRouteData会得到一个封装了路由信息的RouteData对象,而GetVirtualPath则会生成一个URL,该URL被封装成VirtualPathData对象进行返回。

  那通过以上猜测,我们知道了路由系统物理文件路径添加到一个全局的路由表中,并传入给定的参数调用同名方法去找到一个与指定请求URL相匹配的路由对象,并返回相应的RouteData和VirtualPathData对象。

最新文章

  1. 安卓通用shell大全
  2. 集合函数COUNT
  3. 移动金融APP分析
  4. python 映射列表 学习
  5. 解决maven仓库有jar包但是maven程序无法下载仓库jar包
  6. thinkphp对文件的上传,删除,下载操作
  7. NOI2013 UOJ122 向量内积
  8. java设计模式-----单例设计模式
  9. Windows2016的 IIS中配置PHP7运行环境
  10. React(三)JSX内置表达式
  11. 算法:时间复杂度+二分查找法(Java/Go/Python)实现
  12. sublime text2 中标签高亮效果BracketHighlighter插件
  13. cxGrid导出Excel货币符号问题
  14. Jmeter_前端RSA加密下的登陆模拟_引用js文件实现
  15. UVa 147 Dollars(完全背包)
  16. 总结 vb与数据库的连接方法
  17. UVA 10891 Game of Sum(区间DP(记忆化搜索))
  18. Flask中的session和cookie以及日志
  19. java中静态代码块的用法和static用法(转)
  20. ActiveRecord初始化,可以实现jfinal系统启动完成后,再建立数据库连接

热门文章

  1. 關於Validform 控件 值得注意的地方
  2. php字符串函数(1)
  3. Java对数组对象进行排序
  4. (转)C# NameValueCollection集合
  5. spring配置文件中属性mappingLocations、mappingDirectoryLocations
  6. Angularjs总结(三)摸态框的使用
  7. OSI 7层模型
  8. php接口开发--复制缩减Codeigniter的车轮
  9. PHP开发APP接口---返回数据的封装类
  10. prepare—Article【准备篇】之SSH_tool#PuTTY