之前在探讨ASP.NET  MVC的路由时,无意发现原本ASP.NET也有路由机制的。在学习MVC的路由时觉得这部分的资料不太多,不怎么充实(也许是我不懂得去看微软的官方文档)。后来也尝试一下ASP.NET的路由,本文也算是阅读了蒋金楠和重典两位老师后写的读书笔记吧!

  路由机制最显著的一个效果就是实现URL和物理文件的分离。这个分离了之后有三个好处:更灵活,更好的可读性,SEO更友好。

具体是这样的:灵活在于文件的路径有了更改(例如放到了一个新的文件夹里面),那就得把所有涉及到那个文件的URL都改一遍,懒一点的就Ctrl+H。如果用了路由映射的话,只需要在一个地方改就可以了,简洁省事;更好的可读在于传统的URL在传参的时候,都会在问号“?”后面都会以[参数名]=[参数值]的形式一个个的连接起来,就像这样子

Http://127.0.0.1:8083/WebForm1.aspx?param1=parameterValue1&param2=parameterValue2

但是在路由机制下的URL会变得比较简洁明了

Http://127.0.0.1:8083/WebForm1/parameterValue1/parameterValue2(路由的模式暂不提);

SEO友好这点我无法举例子了,呵呵!

  下面则举一个简单的例子来演示如何利用这个路由机制来实现URL与物理文件的分离。

在MVC的项目的Global.asax文件中,路由的定义都放在了一RegisterRoutes(RouteCollection routes)的静态方法里头,这个方法在Application_Start()调用。而在ASP.NET里面也是类似,路由的定义都得在Application_Start()里面完成,代码如下

  1. protected void Application_Start(object sender, EventArgs e)
  2. {
  3. RouteValueDictionary defaultParam = new RouteValueDictionary { { "param1", "*" }, { "param2", "*" } };
  4. RouteTable.Routes.MapPageRoute("default", "WebForm1/{param1}/{param2}", "~/WebForm1.aspx", true, defaultParam);
  5. }

这里主要添加路由的是调用MapPageRoute方法,里面的参数大致跟MVC下添加路由的类似,也是包含了路由的名称,路由的模式,物理文件名等等,在这里我给两个参数都定义了默认值,param1和param2都是“*”,其实也可以定义其他的约束,例如参数值要符合某种格式要求,至于MapPageRoute方法的其他重载,罗列如下

  1. public Route MapPageRoute(string routeName, string routeUrl, string physicalFile);
  2. public Route MapPageRoute(string routeName, string routeUrl, string physicalFile, bool checkPhysicalUrlAccess);
  3. public Route MapPageRoute(string routeName, string routeUrl, string physicalFile, bool checkPhysicalUrlAccess, RouteValueDictionary defaults);
  4. public Route MapPageRoute(string routeName, string routeUrl, string physicalFile, bool checkPhysicalUrlAccess, RouteValueDictionary defaults, RouteValueDictionary constraints);
  5. public Route MapPageRoute(string routeName, string routeUrl, string physicalFile, bool checkPhysicalUrlAccess, RouteValueDictionary defaults, RouteValueDictionary constraints, RouteValueDictionary dataTokens);

在这个例子中需要一个位于根目录下的aspx页面,名为WebForm1.aspx,它的load事件绑定的方法如下

  1. protected void Page_Load(object sender, EventArgs e)
  2. {
  3. StringBuilder sb = new StringBuilder();
  4. string temp=this.RouteData.Values["param1"].ToString();
  5. sb.AppendFormat("Param1:{0}<br/>", temp==""?"*":temp);
  6. temp = this.RouteData.Values["param2"].ToString();
  7. sb.AppendFormat("Param2:{0}<br/>", temp == "" ? "*" : temp);
  8. Response.Write(sb.ToString());
  9. }

生成之后,键入不同的URL效果如下

  其实添加这个路由的不光只是通过MapPageRoute方法,这里还有另一种方式,通过实现IHttpHandler接口和IRouteHandler。还是从头开始说说吧,RouteTable的Routes属性实际上是一个RouteCollection类型的实例,由于它是一个集合类型的,可以通过调用它的Add方法来添加一个路由Route的实例。这个Route的实例就包括了URL的模式和一些处理规则(包括请求中对应哪个物理文件等等。)

涉及到的类的类图如下

这里定义了两个类,一个MyHttpHandler,实现IHttpHandler接口;另一个MyRouteHandler,实现IRouteHandler接口。Route的构造函数可传入一个IRouteHandler的参数,传参时就用到自己定义的MyRouteHandler类。而这个IRouteHandler的成员里头有个GetHttpHandler(RequestContext requestContext)方法,这个方法就是获取一个实现IHttpHandler的类的实例,这里就返回就是自己定义的MyHttpHandler的实例。至于MyHttpHandler类里面,有一个virtual void ProcessRequest(HttpContext context)的虚方法,这个虚方法就是实际请求一个虚拟路径上的处理程序。

这里粘一下代码,主要是两个类的定义

  1. public class MyHttpHandler : IHttpHandler
  2. {
  3.  
  4. public RequestContext RequestContext { get; private set; }
  5.  
  6. public MyHttpHandler(RequestContext context)
  7. {
  8.  
  9. this.RequestContext = context;
  10.  
  11. }
  12.  
  13. #region IHttpHandler 成员
  14.  
  15. public virtual void ProcessRequest(HttpContext context)
  16. {
  17. //这里调用的文件的物理路径,如果文件的路径有改动的话,统一在这里改就可以了
  18. context.Server.Execute("/" + RequestContext.RouteData.Values["page"]);
  19. }
  20.  
  21. public bool IsReusable
  22. {
  23.  
  24. get { return false; }
  25.  
  26. }
  27.  
  28. #endregion
  29.  
  30. }

MyHttpHandler类的

  1. public class MyRouteHandler : IRouteHandler
  2. {
  3.  
  4. #region IRouteHandler 成员
  5.  
  6. public IHttpHandler GetHttpHandler(RequestContext requestContext)
  7. {
  8.  
  9. return new MyHttpHandler(requestContext);
  10.  
  11. }
  12.  
  13. #endregion
  14.  
  15. }

MyRouteHandler类的

在Application_Start(object sender, EventArgs e)里面只需添加一行代码

  1. RouteTable.Routes.Add(new Route("{param1}/{param2}/{page}", new MyRouteHandler()));

WebForm1.aspx的代码不需要作任何更改,用http://localhost:1144/122343/abcdef/WebForm1.aspx发出请求,结果还是一样

不过这里的参数不能为空了,以为没有设默认值。

在前面罗列MapPageRoute方法的重载时也发现,URL上面的参数可以给参数设定默认值,对参数的格式进行限制,下面则尝试尝试。无论是默认值还是格式约束,都要使用RouteValueDictionary这个类。

  1. protected void Application_Start(object sender, EventArgs e)
  2. {
  3. RouteValueDictionary defaultParam = new RouteValueDictionary { { "param1", "" }, { "param2", "" } };//param1和param2的默认值都是0
  4. RouteValueDictionary constraint = new RouteValueDictionary { { "param2", @"^\d+$" } };//param2要是一个正整数
  5. RouteTable.Routes.MapPageRoute("default", "WebForm1/{param1}/{param2}", "~/WebForm1.aspx", true, defaultParam, constraint);
  6. }

各个URL和结果如下列表所示

请求URL

结果

http://localhost:1144/WebForm1

Param1:0
Param2:0

http://localhost:1144/WebForm1/sdfb

Param1:sdfb
Param2:0

http://localhost:1144/WebForm1/sdfb/343

Param1:sdfb
Param2:343

http://localhost:1144/WebForm1/sdfb/sfe

HTTP 404 错误
无法找到资源。

  如果用IHttpHandler接口和IRouteHandler的话,则需要在Application_Start方法里面做一下改动

  1. protected void Application_Start(object sender, EventArgs e)
  2. {
  3. RouteValueDictionary defaultParam = new RouteValueDictionary { { "param1", "" }, { "param2", "" } };
  4. RouteValueDictionary constraint = new RouteValueDictionary { { "param2", @"^\d+$" } };
  5. RouteTable.Routes.Add(new Route("{page}/{param1}/{param2}",defaultParam,constraint, new MyRouteHandler()));
  6. }

结果跟上面表格的一样。

  对于上面使用MapPageRoute方法的这种情况来说,如果按照上面的路由设置,那么如果按照之前文件路径那样去请求的话,照样能访问到指定的页面。例如就上面一直使用的WebForm1.aspx,按照这个http://localhost:1144/WebForm1.aspx URL去请求的话,也能访问到WebForm1.aspx,但是有差别的是我们的路由它不作任何处理,在路由里设置的默认值根本没有生效,结果如下:

Param1:*

Param2:*

如果要让这种URL也要路由的话,则需要设置一个属性

  1. RouteTable.Routes.RouteExistingFiles = true;

顾名思义,它表明了是否要对一个存在的文件进行路由。它其实是RouteCollection类的一个属性,对于整个站点的路由来说,它是一个全局属性,默认值为false。当把它设成ture之后,用回http://localhost:1144/WebForm1.aspx进行请求,得出的结果如下:

Param1:WebForm1.aspx

Param2:0

也就是说路由生效了,WebForm1.aspx被看作是参数1,而不是一个物理文件的文件名了,0则是参数2的默认值。

  现在所有URL都会经过路由处理,那么js,css,图片等文件引用都会被路由。例如在页面上添加这个

  1. <img src="b6126b3468327b5c251f143c.jpg" width="300" height="500" />

结果只能是这样

这时需要对部分文件取消路由,

  1. RouteTable.Routes.Ignore("{filename}.jpg/{*pathInfo}");

图片就可以出来了,吾王归来

这种方式只能对对一种文件进行取消,js的要设置,css要设置,png要设置,gif要设置……可是MVC那里是Ignore了这种"{resource}.axd/{*pathInfo}"就可以了,到这里就是换成这样子

  1. RouteTable.Routes.Ignore("{resource}.axd/{*pathInfo}");

可是吾王又不见了,为啥????

  

  这个路由机制还有另一个用途,就是根据路由来构造新的URL,这个构造主要是利用一个方法GetVirtualPath,这个方法RouteCollection有,Route也有。不同的是,调用RouteCollection的GetVirtualPath时,它会遍历整个集合中所有Route对象,逐个对象去调用该Route对象自身的GetVirtualPath方法,直到返回值不为null为止,如果到最终都是null的,那只能返回null。

当我们定义这样的路由

  1. RouteValueDictionary defaultParam = new RouteValueDictionary { { "param1", "" }, { "param2", "" } };
  2. RouteValueDictionary constraint = new RouteValueDictionary { { "param2", @"^\d+$" } };
  3.  
  4. RouteTable.Routes.MapPageRoute("default", "{param1}/{param2}", "~/WebForm1.aspx", true, defaultParam, constraint);

在WebForm1.aspx的与Load时间绑定的方法里面添加以下代码

  1. RouteData routeData = new RouteData();
  2. routeData.Values.Add("param1", "abc");
  3. routeData.Values.Add("param2", "");
  4.  
  5. RouteValueDictionary values = new RouteValueDictionary();
  6. values.Add("param1", "efg");
  7. values.Add("param2", "");
  8.  
  9. Response.Write(RouteTable.Routes.GetVirtualPath(null, null).VirtualPath + "<br/>");
  10. Response.Write(RouteTable.Routes.GetVirtualPath(Request.RequestContext, null).VirtualPath + "<br/>");
  11. Response.Write(RouteTable.Routes.GetVirtualPath(Request.RequestContext, values).VirtualPath + "<br/>");

当我们以http://localhost:1144/abc/123请求时,得出的三个URL分别是

/

/abc/123

/efg/456

  从上面代码看出,第一次调用时是没有传RequestContext,也没有提供路由的参数,得出的URL是“/”;第二次调用时只传了当前的RequestContext,没有提供路由参数,得出的URL跟当前的一样,是“/abc/123”;第三次RequestContext和路由参数都传了,路由参数是param1是abc,param2是123,得出的URL是“/efg/456”。由此可见,当传入了路由参数时,生成的URL肯定是按照路由参数生成的;当没传路由参数且只传RequestContext时,生成的URL是按照RequestContext的路由参数来生成的;当什么也没传的时候,就只能生成所有参数为空的URL。即对于生成URL来说,路由参数比RequestContext优先级更高。

  其实这个构造URL的有什么作用我还不清楚,先记着留个印象,到后来万一用上也可以留个底。

  这篇文章呐其实在两个月之前就起草了,由于各种原因搁置了下来,现在重新写一下。曾经想过写一系列的有关APS.NET MVC的文章,可惜了解的少,能写的更少。这篇文章里面不足的肯定很多,希望各位多多指点,谢谢!

最后附上一些比较有参考价值的文章

ASP.NET MVC路由扩展:路由映射

ASP.NET的路由系统:URL与物理文件的分离

ASP.NET的路由系统:根据路由规则生成URL

ASP.NET的路由系统:路由映射

System.Web.Routing入门及进阶 上篇

System.Web.Routing入门及进阶 下篇

小弟这篇文章来源于上面罗列文章大部分内容,如果冒犯了的,小弟把这篇文章撤出园子首页吧!

ASP.NET的路由的更多相关文章

  1. ASP.NET MVC 路由(一)

    ASP.NET MVC路由(一) 前言 从这一章开始,我们即将进入MVC的世界,在学习MVC的过程中在网上搜索了一下,资料还是蛮多的,只不过对于我这样的初学者来看还是有点难度,自己就想看到有一篇引导性 ...

  2. ASP.NET MVC 路由(二)

     ASP.NET MVC路由(二) 前言 在上一篇中,提及了Route.RouteCollection对象的一些信息,以及它们的结构所对应的关系.按照处理流程走下来还有遗留的疑问没有解决这个篇幅就来讲 ...

  3. ASP.NET MVC 路由(三)

    ASP.NET MVC路由(三) 前言 通过前两篇的学习会对路由系统会有一个初步的了解,并且对路由系统中的Url规则有个简单的了解,在大家的脑海中也有个印象了,那么路由系统在ASP.NETMVC中所处 ...

  4. ASP.NET MVC 路由(四)

    ASP.NET MVC路由(四) 前言 在前面的篇幅中我们讲解路由系统在MVC中的运行过程以及粗略的原理,想必看过前面篇幅的朋友应该对路由有个概念性的了解了,本篇来讲解区域,在读完本篇后不会肯定的让你 ...

  5. ASP.NET MVC 路由(五)

    ASP.NET MVC 路由(五) 前言 前面的篇幅讲解了MVC中的路由系统,只是大概的一个实现流程,让大家更清晰路由系统在MVC中所做的以及所在的位置,通过模糊的概念描述.思维导图没法让您看到路由的 ...

  6. .NET/ASP.NET Routing路由(深入解析路由系统架构原理)

    阅读目录: 1.开篇介绍 2.ASP.NET Routing 路由对象模型的位置 3.ASP.NET Routing 路由对象模型的入口 4.ASP.NET Routing 路由对象模型的内部结构 4 ...

  7. Asp.Net MVC 路由 - Asp.Net 编程 - 张子阳

    http://cache.baiducontent.com/c?m=9d78d513d98316fa03acd2294d01d6165909c7256b96c4523f8a9c12d522195646 ...

  8. .NET/ASP.NET Routing路由(深入解析路由系统架构原理)http://wangqingpei557.blog.51cto.com/1009349/1312422

    阅读目录: 1.开篇介绍 2.ASP.NET Routing 路由对象模型的位置 3.ASP.NET Routing 路由对象模型的入口 4.ASP.NET Routing 路由对象模型的内部结构 4 ...

  9. [Buffalo]ASP.NET MVC路由映射

    Asp.Net的路由系统旨在通过注册URl模版与物理文件之间的映射进而实现请求地址与文件路径之间的分离,但对于Asp.Net Mvc应用来说,请求的目标却是定义在某个Controller类型中的Act ...

随机推荐

  1. 解决tkinter在windows上没有正确安装的问题

    问题 Can't find a usable tk.tcl in the following directories: 解决方法 加两个环境变量,在我的机器上是这样的 TCL_LIBRARY=D:\d ...

  2. mssql中sp_executesql的用法

    上图:

  3. Apache shiro之身份验证(登陆)流程

    从张开涛blog学习后整理:http://jinnianshilongnian.iteye.com/blog/2018398 上图中的类和接口都可以继承和实现来个性化自己的实现. 其中重点看一下Mod ...

  4. android 抓包 使用 tcpdmp + Wireshark

         下载地址tcpdump: http://www.androidtcpdump.com/      使用su用户, 给/system/可写的权限 mount -o remount,rw -t ...

  5. 一种线程安全的handle

    对象引用的正确性在多线程环境下是一个复杂的问题,请参考,处理由引用计数引起的泄漏.简单来说,我们应该尽量减少使用强引用,否则将有可能产生[处理由引用计数引起的泄漏]一文中描述的难以察觉的内存泄漏问题. ...

  6. 严重: Exception starting filter struts2解决方法!

    转自:http://blog.knowsky.com/260578.htm 问题出现: 严重: Exception starting filter struts2java.lang.ClassNotF ...

  7. Windows Store 开发总结——文件操作

    1.读取Isolated Storage 每个Metro程序都有三个文件夹:Local,Roaming,Temp.每个文件夹的访问方法都是相同的. Local用于将数据存储在本地,这是程序特定的文件夹 ...

  8. db2安装及卸载

    创建用户和组: #创建组信息 groupadd -g db2iadm1 groupadd -g db2fadm1 groupadd -g dasadm1 #创建用户信息 useradd -u -g d ...

  9. iOS开发之时间格式的转化

    在开发iOS程序时,有时候需要将时间格式调整成自己希望的格式,这个时候我们可以用NSDateFormatter类来处理. 例如:如何将格式为“12-May-14 05.08.02.000000 PM” ...

  10. 解决“iOS 7 app自动更新,无法在app中向用户展示更新内容”问题

    转自cocoachina iOS 7能在后台自动app,这对开发者来说和用户都很方便,但是还是有一些缺点.用户不会知道app本次更新的内容,除非他们上到app的App Store页面去查看.开发者也会 ...