ASP.NET MVC重写了ASP.NET管道HttpModule和处理程序HttpHandler。MVC自定义了MvcHandler实现了Controller的激活和Action的执行。但是在请求到达这里之前把Http请求Url解析到正确的Controller和Action上,是通过自定义HttpModule来实现的,这个自定义的HttpModule就是Url路由系统UrlRoutingModule。
1:UrlRoutingModule拦截请求的Url,对Url进行解析,得到以Controller和Action为核心的路由数据。
2:UrlRoutingModule的好处:
1:灵活性,Url无需对应磁盘上的物理文件
2:可读性,可以设置自定义的Url规则。
3:SEO优化,对Url进行有效的优化,更容易被搜索收录。
在以前IIS有一个Url重写模块,Url被IIS处理之前,根据其规则,重定向到物理文件。这个重写是IIS级别的。而MVC路由系统,它是ASP.NET的一部分,是通过托管代码实现的。
3:Page类有一个公开的属性Page. RouteData,System.Web.Routing.RouteData表示通过路由系统对当前请求进行解析得到的路由数据。RouteData.Values属性表示存储路由变量的字典,可以从中取出路由变量的值。
var defaults = new RouteValueDictionary{ { "name", "*" }, { "id", "*" } };
RouteTable.Routes.MapPageRoute("", "employees/{name}/{id}", "~/EmployeePage.aspx", true, defaults);
上面注册路由变量name,id,在Page页面可以动过RouteData取出
string name = Page.RouteData.Values["name"].ToString()
string id = Page.RouteData.Values["id"].ToString()
4:Route和RouteTable
-->一个Web具有一个全局的路由表,该表是通过System.Web.Routing.RouteTable的静态只读字段Routes表示
-->System.Web.Routing.RouteTable.Routes返回一个System.Routing.RouteCollection的集合。表示整个Web的全局路由表。
-->调用RouteCollection的MapPageRoute方法,将物理文件映射到一个URL模版上,这个过程是基于指定的URL模版创建一个路由对象并添加到全局路由表中。
-->System.Web.Routing.Route是抽象类RouteBase唯一的直接子类,基于模版创建的路由匹配规则就定义在Route中。
-->Route匹配请求的地址是否与Url模版匹配?首先将Url通过分隔符/进行拆分,拆分成多个段,每个段又分为变量和文本。变量有{}包围,文本是单纯的文本。检查文本的内容是否一致,以及变量的数量是否一致。
-->Route属性Constraints为定义在Url模版中的变量以正则表达式的形式座一些限定。属性Constraints类型为RouteValueDictionary,Key和Value分别表示变量名和限定的表达式。定义了Constraints,除了验证Url模版的段,还需通过正则表达式的验证。
-->Route属性Default为变量定义了默认值。Default类型为RouteValueDictionary
-->Route属性DataTokens通过路由传递到处理程序,未使用的值。DataTokens的类型为RouteValueDictionary。
-->RouteTable.Routes.RouteExistingFiles,表示是否对已存在的物理文件使用路由。
-->通量匹配,可以采用通量匹配来匹配Url最后的部分,可以包含多个段。{*变量}
RouteTable.Routes.MapPageRoute("", "employees/{name}/{*id}", "~/EmployeePage.aspx", true, defaults);
/employees/李四/002/1212
那么变量id,能够匹配到002/1212
-->System.Web.Routing.RouteTable.Routes,通常有两个方法来注册路由
System.Web.Routing.RouteTable.Routes.MapPageRoute() 注册物理文件与Url模版之间的映射,也就是给System.Web.Routing.RouteCollection集合添加一个Route对象
System.Web.Routing.RouteTable.Routes.Ignore() 注册让路由系统忽略的Url模版。
5:使用System.Web.Routing.RouteTable.Routes.MapPageRoute()注册物理文件与Url模版之间的映射。
//对已存在的物理文件也使用路由
RouteTable.Routes.RouteExistingFiles=true;
//Route对象的Default属性,指定路由变量的默认值。
var defaultW = new RouteValueDictionary { { "areacode", "010" }, { "days", 2 } };
//Route对象的Constraints属性,使用正则表达式约束路由变量。
var constraints = new RouteValueDictionary { { "areacode", @"0d{2,3}" }, { "days", @"[1-3]{1}" } };
//Route对象的DataTokens属性,给处理程序传递值。
var dataTokens = new RouteValueDictionary { { "defaultCity", "Beijing" }, { "defaultDays", 2 } };
RouteTable.Routes.MapPageRoute("Weather", "Weather/{areacode}/{days}", "~/Weather.aspx", false, defaultW, constaints, dataTokens);
6:使用System.Web.Routing.RouteTable.Routes.Ignore()忽略Url模版。Routes.Ignore()方法需要放在注册路由方法Routes.MapPageRoute()前面。否则不起作用。
启用了RouteTable.Routes.RouteExistingFiles=True,会对所有的请求都按照路由表进行解析。那么针对CSS文件,JavaScript文件也进行了解析。那么就要使用Routes.Ignore()忽略静态文件的脚本
//对已存在的物理文件也使用路由
RouteTable.Routes.RouteExistingFiles=true;
RouteTable.Routes.Ignore("{filename}.css/{*pathInfo}");
7:使用System.Web.Routing.RouteTable.Routes.add(new Route())将路由添加到全局路由表中。它的作用和System.Routing.RouteTable.Routes.MapPageRoute()是一样的
//创建Route对象
Route route = new Route("{areacode}/{days}",
new RouteValueDictionary { { "areacode", "010" }, { "days", 2 } },
new RouteValueDictionary { { "areacode", @"0d{2,3}" }, { "days", @"[1-3]{1}" }, { "httpMethod", new HttpMethodConstraint("GET") } },
new PageRouteHandler("~/Weather.aspx", true));
//将Route对象添加到全局路由表中
RouteTable.Routes.Add(route);
8:根据注册的路由规则生成相应的Url,使用RouteTable.Routes.GetVirtualPath(RequestContext,RouteValueDirectionary).ViretualPath;
RouteData r = new RouteData();
r.Values.Add("area","024");
r.Values.Add("days", 2);
Response.Write(RouteTable.Routes.GetVirtualPath(null,null).VirtualPath);
Response.Write("<br />...................");
RequestContext requestContext = new RequestContext();
requestContext.HttpContext = new HttpContextWrapper(HttpContext.Current);
requestContext.RouteData = r;
RouteValueDictionary values = new RouteValueDictionary { { "area", "011" }, { "days", 2 } };
Response.Write(RouteTable.Routes.GetVirtualPath(requestContext, null).VirtualPath);
Response.Write("<br />...................");
Response.Write(RouteTable.Routes.GetVirtualPath(requestContext, values).VirtualPath);
Response.Write("<br />...................");