asp.net MVC 路由机制 Route
1:ASP.NET的路由机制主要有两种用途:
-->1:匹配请求的Url,将这些请求映射到控制器
-->2:选择一个匹配的路由,构造出一个Url
2:ASP.NET路由机制与URL重写的区别
-->Url重写关注的是将一个Url映射到另一个Url。
路由机制关注的是将Url映射到资源上。资源不一定是一个物理页面,可以是类中的方法。
-->Url重写只能用于传入的请求Url
路由机制可以匹配传入的Url,也能够生成一个Url
-->Url重写大多是IIS级别的,是IIS的一个组件
路由机制是HttpModule级别,可以用代码进行良好地控制
3:路由Url
RouteTable.Routes.MapRoute("simple","{first}/{second}/{third}');
第一个参数是路由名称,第二个参数是路由模式。
路由系统会将客户端请求的Url,依照路由模式进行解析,并将其解析到RouteValueDictionary实例的键/值中,存储到RouteData中,可以通过RequestContext访问RouteValueDictionary中的值。key就是路由模式中德参数名称,值为请求url中的值。例如:
请求的url为:/abc/display/123,那么解析到RouteData中的数据为:first="abc",second="display",third="123"
4:MVC中特殊的Url参数名称-{controller}和{action}
因为在MVC中,Url都会映射到控制器上的一个方法上,所以MVC框架需要使用一些特定的参数名称{controller}{action}。{controller}参数用来实例化一个控制器类。按照约定优先的惯例,MVC将字符串Controller添加到{controller}参数值的后面,得到一个类名,然后根据这个类名查找实现了System.Web.Mvc.IController接口的类型,完成实例化。除了{controller}{action}之外,其他的参数当作控制器action方法的参数来处理。
RouteTable.Routes.MapRoute("simple","{controller}/{action}/{id}');
/abc/display/123请求会实例成abcController的控制器类,调用其中的display()方法,同时将123传递给display()方法的参数id
5:路由中的字面值
Url段中允许字面值和参数混合在一起。它仅有的限制就是不允许有两个连续Url的参数
{language}-{country}/{controller}/{action}
{controller}.{action}.{id}
这些都是合法的。
6:路由的默认值
为Routes.MapRoute()方法传递一个默认值的字典。我们可以使用简明的语法来定义字典,MapRoute()方法会在底层将简明的语法new {controller="Home",action="Index",id=UrlparameterOptional}转换成一个RouteValueDirectionary的一个实例。
-->1默认值对于Url参数位置十分重要,只有为当前参数后面的每一个参数都定义了默认值,路由才会采用当前参数的默认值,{controller}/{action}/{id},如果我们为{action}提供了默认值,没有为{id}提供默认值,那么效果与不给{action}默认值是一样的。下面例子:
RouteTable.Routes.MapRoute("simple1","{controller}/{action}/{id}",new {action="index"});
RouteTable.Routes.MapRoute("simple2","{controller}/{action}");
/abc/display这个Url只能匹配到simple2,因为只有为当前参数后面的每一个参数都定义了默认值,路由才会采用当前参数的默认值。
-->2任何带有字面量的Url段在匹配请求的URL时,都禁止省略任何参数值。
RouteTable.Routes.MapRoute("simple","{controller}-{action}",new {action="Index"});
它并不能匹配/abc-的请求,因为字面量-后面的参数{action}被省略了。
7:路由约束
请求Url段的数量与路由模式中定义的参数个数能够匹配上,那么路由就能匹配这个url,但是我们需要对Url有更多的控制,就需要使用路由约束。
为Routes.MapRoute()方法传递一个正则表达式约束的字典。我们可以使用简明的语法来定义字典,在方法的内部使用Regex类,将其转换成RouteValueDirectionary类型的对象。
RouteTable.Routes.MapRoute("simple","{controller}/{action}/{id}",new {controller="Home",action="Index",Id=""},new {controller=@"\w+",action=@"\w+",id=@"\d+"});
-->自定义路由约束
除了正则表达式约束之外,我们可以实现IRouteConstraint接口,实现接口的Match()方法
系统已经为我们创建了一个实现IRouteConstraint接口的HttpMethodConstraint类,约束路由只能匹配指定的Http方法。
RouteTable.Routes.MapRoute("default","{controller}/{action}/{id}",null,new {httpMethod=HttpMethodConstraint("GET")}); 这个路由只能匹配Get请求。
8:路由名称
-->1:在路由系统匹配请求的Url的时候,可以不需要用到路由名称。但是在生成一个Url的时候,就需要一个已经定义的路由名称,按照这个路由定义的规则,生成一个Url。在生成Url的时候,对路由选择进行精确控制。
routes.MapRoute(
name: "Default",
url: "{controller}/{action}/{id}",
defaults: new { controller = "Home", action = "Index", id = UrlParameter.Optional }
);
routes.MapRoute(
name: "Test",
url: "code/p/{action}/{id}",
defaults: new {Controller="Home",action="index",id=UrlParameter.Optional }
);
@Html.RouteLink("Test", "Test", new { action = "indexTest",id=123 })
选择Test的路由,生成的Url是code/p/indexTest/123
@Html.RouteLink("Default", "Default", new { controller = "homeTest", action = "indexTest",id=123 })
先择Default路由,生成的Url是homeTest/indexTest/123
-->2:在使用路由创建Url的时候,如果提供的路由有多余的,将被添加到url的后面,当作URL参数。
@Html.RouteLink("Test", "Test", new { action = "indexTest",id=123,para=456 })
选择Test的路由,生成的Url是code/p/indexTest/123?para=456
9:MVC区域
区域允许我们将模型,视图和控制器分成单独的功能节点。我们可以将大型复杂的网站分成若干个节点,方便管理。每一个区域都有单独的路由系统。
我们可以通过为每一个区域创建一个类,来配置区域路由。这个类要派生自AreaRegistration类,重写其中的AreaName和RegisterArea成员。在Global.asax文件中,调用AreaRegistration.RegisterAllAreas()方法,就会调用每一个区域的RegisterArea()
-->区域路由的冲突。
当在两个区域中,当有两个相同名称的控制器,那么当前传入的请求匹配没有指定名称空间的路由时,系统会抛出一个异常
9.1:名称空间来区分控制器的优先顺序
当输入的一个Url请求与一条路由匹配时,如果有多个同一名称的控制器,就会出现异常。我们可以在路由注册的时候,对某些名称空间制定优先级。
routes.MapRoute("myRoute","{controller}/{action}/{id}/{*catchall}",new{controller="Home",action="Index",id=UrlParameter.Optional},new[]{"MyNamespace"});
9.2:禁用后备的名称空间
可以告诉MVC路由系统,只查看指定的名称空间。在这个指定的名称空间下,如果找不到控制器,就停止搜素。
Route myRoute=routes.MapRoute("myRoute","{controller}/{action}/{id}/{*catchall}",new{controller="Home",action="Index",id=UrlParameter.Optional},new[]{"MyNamespace"});
myRoute.DataTokens["UserNamespaceFallback"]=fasle;
myRoute.DataTokens["UserNamespaceFallback"]=fasle此设置会传递到控制器工厂。
10:通量匹配catch-all
catch-all允许我们匹配任意数量的段的Url
RouteTable.Routes.MapRoute("default","{query/{query-name}}/{*extrastuff}")
/query/select/a/b/c 参数{extrastuff}=a/b/c
/query/select/ 路由仍能匹配{extrastuff}=""
路由Url在与传入的请求匹配时,它的字面量与请求精确匹配的,而参数是贪婪匹配的。每个Url参数都尽可能多地匹配文本。
RouteTable.Routes.MapRoute("defalut","{filename}.{ext}")
/asp.net.mvc.xml {filename}=asp.net.mvc {ext}=xml
因为Url参数是贪婪匹配所以{filename}尽可能匹配多的文本,但需要给{ext}留下匹配的空间。
11:路由机制忽略请求的匹配
-->1:StopingRoutingHandler,在创建Route对象的时候,选择处理程序
为RouteCollection添加路由Route,如果选择StopingRoutingHandler来创建,那么就会忽略这个Route匹配的请求
StopingRoutingHandler会忽略匹配请求的Url,将这个请求传递给标准的HTTP处理程序
MvcRouteHandler会创建MVC路由对象,将这个请求传递给MvcRoute处理程序
PageRouteHandler会创建ASP.NET路由对象,将这个请求传递给PageRoute处理程序
Route axdRoute = new Route("{resource}.axd/{*pathInfo}", new StopRoutingHandler());
routes.Add(axdRoute);
Route mvcRoutes = new Route("{resource}.axd/{*pathInfo}",new MvcRouteHandler());
Route pageRoutes = new Route("{resource}.axd/{*pathInfo}", new PageRouteHandler("~/Weather.aspx", true));
-->2:使用RouteCollection的扩展方法IgnoreRoute()忽略请求的Url
RouteTable.Routes.IgnoreRoute("{resource}.axd/{*pathInfo}");
12:路由的测试
我们可以使用RouteDebugger来测试路由
13:路由生成Url
路由可以匹配请求的Url,那么也可以生成一个Url,这是一个完整的双向系统。
-->1:调用RouteCollection.GetVirtualPath(),传递一个RequestContext对象,一个包含值得字典,一个Url路由名称。
-->2:检查路由的参数是否和字典提供的值相匹配,并检查默认值,约束
-->3:依据字典提供的值,生成一个Url
-->4:溢出参数,如果字典中有多余的参数,那么这些参数会附加到Url,当作查询字符串参数
@html和@url生成链接的方法,都是调用RouteCollection.GetVirtualPath()方法。
14:路由请求管道UrlRoutingModule
-->1:UrlRoutingModule使用在RouteTable中注册的路由匹配当前请求。
-->2:匹配成功,路由模块从匹配成功的路由对象Route中获取IRouteHandler接口对象,一般MvcRouteHandler类就是这个实现。
-->3:路由模块调用IRouteHandler接口的(就是MvcRouteHandler的实例)GetHandler()方法,返回用来处理请求的IHttpHandler对象,一般是MvcHandler。
-->4:调用实现IHttpHandler接口的HTTP处理程序对象的ProcessRequest()方法,将要处理的请求传递给她。
-->5:ASP.NETMVC中,MvcRouteHandler类就是IRouteHandler的实现,MvcRouteHandler返回一个实现了IHttpHandler接口的MvcHandler对象。MvcHandler对象用来实例化控制器类,调用控制器上面的方法。
15:路由数据RouteData
RouteCollection继承自RouteBase,RouteBase定义了GetVirtualPath()方法和GetRouteData()方法,GetRouteData()方法,返回包含路由数据的字典对象RouteData