一、为什么需要路由优先级

大家都知道我们在Asp.Net MVC项目或WebApi项目中注册路由是没有优先级的,当项目比较大、或有多个区域、或多个Web项目、或采用插件式框架开发时,我们的路由注册很可能不是写在一个文件中的,而是分散在很多不同项目的文件中,这样一来,路由的优先级的问题就突显出来了。

比如: App_Start/RouteConfig.cs中

routes.MapRoute(
name: "Default",
url: "{controller}/{action}/{id}",
defaults: new { controller = "Home", action = "Index", id = UrlParameter.Optional }
);

Areas/Admin/AdminAreaRegistration.cs中

context.MapRoute(
name: "Login",
url: "login",
defaults: new { area = "Admin", controller = "Account", action = "Login", id = UrlParameter.Optional },
namespaces: new string[] { "Wenku.Admin.Controllers" }
);

假如是先注册上面那个通用的default路由,再注册这个login的路由,那么无论怎么样,都会先匹配第一个满足条件的路由,也就是第两个路由注册是无效的。
造成这个问题的原因就是这两个路由注册的顺序问题,而Asp.Net MVC及WebApi中注册路由都没有优先级这个概念,所以今天我们就是要自己实现这个想法,在注册路由时加入一个优先级的概念。

二、解决思路

1、先分析路由注册的入口,比如我们新建一个mvc4.0的项目

public class MvcApplication : System.Web.HttpApplication
{
protected void Application_Start()
{
AreaRegistration.RegisterAllAreas(); WebApiConfig.Register(GlobalConfiguration.Configuration);
FilterConfig.RegisterGlobalFilters(GlobalFilters.Filters);
RouteConfig.RegisterRoutes(RouteTable.Routes);
}
}

Mvc路由的注册入口有两个:
    a. AreaRegistration.RegisterAllAreas();                            注册区域路由

b. RouteConfig.RegisterRoutes(RouteTable.Routes);          注册项目路由

WebApi路由注册入口有一个:

    WebApiConfig.Register(GlobalConfiguration.Configuration);  注册WebApi路由

2、注册路由的处理类分析

    AreaRegistrationContext

RouteCollection

HttpRouteCollection

注册路由时主要是由这三个类来注册处理路由的。

3、路由优先级方案

    a、更改路由的注册入口

b、自定义一个路由的结构类RoutePriority及HttpRoutePriority,这两个类下面都有Priority这个属性

c、自定一个RegistrationContext来注册路由,注册的对象为上述自定义路由。

d、所有的路由注册完成之后再按优先顺序添加到RouteCollection及HttpRouteCollection中实际生效。

三、具体实现

1、路由定义

public class RoutePriority : Route
{
public string Name { get; set; }
public int Priority { get; set; } public RoutePriority(string url, IRouteHandler routeHandler)
: base(url,routeHandler)
{ }
} public class HttpRoutePriority
{
public string Name { get; set; }
public int Priority { get; set; }
public string RouteTemplate{get;set;}
public object Defaults{get;set;}
public object Constraints{get;set;}
public HttpMessageHandler Handler{get;set;}
}

2、定义路由注册的接口

public interface IRouteRegister
{
void Register(RegistrationContext context);
}

3、定义路由注册上下文类

public class RegistrationContext
{
#region mvc
public List<RoutePriority> Routes = new List<RoutePriority>(); public RoutePriority MapRoute(string name, string url,int priority=)
{
return MapRoute(name, url, (object)null /* defaults */, priority);
} public RoutePriority MapRoute(string name, string url, object defaults, int priority = )
{
return MapRoute(name, url, defaults, (object)null /* constraints */, priority);
} public RoutePriority MapRoute(string name, string url, object defaults, object constraints, int priority = )
{
return MapRoute(name, url, defaults, constraints, null /* namespaces */, priority);
} public RoutePriority MapRoute(string name, string url, string[] namespaces, int priority = )
{
return MapRoute(name, url, (object)null /* defaults */, namespaces, priority);
} public RoutePriority MapRoute(string name, string url, object defaults, string[] namespaces,int priority=)
{
return MapRoute(name, url, defaults, null /* constraints */, namespaces, priority);
} public RoutePriority MapRoute(string name, string url, object defaults, object constraints, string[] namespaces, int priority = )
{
var route = MapPriorityRoute(name, url, defaults, constraints, namespaces, priority);
var areaName = GetAreaName(defaults);
route.DataTokens["area"] = areaName; // disabling the namespace lookup fallback mechanism keeps this areas from accidentally picking up
// controllers belonging to other areas
bool useNamespaceFallback = (namespaces == null || namespaces.Length == );
route.DataTokens["UseNamespaceFallback"] = useNamespaceFallback; return route;
} private static string GetAreaName(object defaults)
{
if (defaults != null)
{
var property = defaults.GetType().GetProperty("area");
if (property != null)
return (string)property.GetValue(defaults, null);
} return null;
} private RoutePriority MapPriorityRoute(string name, string url, object defaults, object constraints, string[] namespaces,int priority)
{
if (url == null)
{
throw new ArgumentNullException("url");
} var route = new RoutePriority(url, new MvcRouteHandler())
{
Name = name,
Priority = priority,
Defaults = CreateRouteValueDictionary(defaults),
Constraints = CreateRouteValueDictionary(constraints),
DataTokens = new RouteValueDictionary()
}; if ((namespaces != null) && (namespaces.Length > ))
{
route.DataTokens["Namespaces"] = namespaces;
} Routes.Add(route);
return route;
} private static RouteValueDictionary CreateRouteValueDictionary(object values)
{
var dictionary = values as IDictionary<string, object>;
if (dictionary != null)
{
return new RouteValueDictionary(dictionary);
} return new RouteValueDictionary(values);
}
#endregion #region http
public List<HttpRoutePriority> HttpRoutes = new List<HttpRoutePriority>(); public HttpRoutePriority MapHttpRoute(string name, string routeTemplate, int priority = )
{
return MapHttpRoute(name, routeTemplate, defaults: null, constraints: null, handler: null, priority: priority);
} public HttpRoutePriority MapHttpRoute(string name, string routeTemplate, object defaults, int priority = )
{
return MapHttpRoute(name, routeTemplate, defaults, constraints: null, handler: null, priority: priority);
} public HttpRoutePriority MapHttpRoute(string name, string routeTemplate, object defaults, object constraints, int priority = )
{
return MapHttpRoute(name, routeTemplate, defaults, constraints, handler: null, priority: priority);
} public HttpRoutePriority MapHttpRoute(string name, string routeTemplate, object defaults, object constraints, HttpMessageHandler handler, int priority = )
{
var httpRoute = new HttpRoutePriority();
httpRoute.Name = name;
httpRoute.RouteTemplate = routeTemplate;
httpRoute.Defaults = defaults;
httpRoute.Constraints = constraints;
httpRoute.Handler = handler;
httpRoute.Priority = priority;
HttpRoutes.Add(httpRoute); return httpRoute;
}
#endregion
}

4、把路由注册处理方法添加到Configuration类中

public static Configuration RegisterRoutePriority(this Configuration config)
{
var typesSoFar = new List<Type>();
var assemblies = GetReferencedAssemblies();
foreach (Assembly assembly in assemblies)
{
var types = assembly.GetTypes().Where(t => typeof(IRouteRegister).IsAssignableFrom(t) && !t.IsAbstract && !t.IsInterface);
typesSoFar.AddRange(types);
} var context = new RegistrationContext();
foreach (var type in typesSoFar)
{
var obj = (IRouteRegister)Activator.CreateInstance(type);
obj.Register(context);
} foreach (var route in context.HttpRoutes.OrderByDescending(x => x.Priority))
GlobalConfiguration.Configuration.Routes.MapHttpRoute(route.Name, route.RouteTemplate, route.Defaults, route.Constraints, route.Handler); foreach (var route in context.Routes.OrderByDescending(x => x.Priority))
RouteTable.Routes.Add(route.Name, route); return config;
} private static IEnumerable<Assembly> GetReferencedAssemblies()
{
var assemblies = BuildManager.GetReferencedAssemblies();
foreach (Assembly assembly in assemblies)
yield return assembly;
}

这样一来就大功告成,使用时只需要在Global.asax.cs文件中修改原注册入口为

public class MvcApplication : System.Web.HttpApplication
{
protected void Application_Start()
{
WebApiConfig.Register(GlobalConfiguration.Configuration);
FilterConfig.RegisterGlobalFilters(GlobalFilters.Filters);
RouteConfig.RegisterRoutes(RouteTable.Routes); Configuration.Instance()
.RegisterComponents()
.RegisterRoutePriority(); //注册自定义路由
}
}

在每个项目中使用只需要要继承自定义路由注册接口IRouteRegister,例如:

public class Registration : IRouteRegister
{
public void Register(RegistrationContext context)
{
//注册后端管理登录路由
context.MapRoute(
name: "Admin_Login",
url: "Admin/login",
defaults: new { area = "Admin", controller = "Account", action = "Login", id = UrlParameter.Optional },
namespaces: new string[] { "Wenku.Admin.Controllers" },
priority:
); //注册后端管理页面默认路由
context.MapRoute(
name: "Admin_default",
url: "Admin/{controller}/{action}/{id}",
defaults: new { area = "Admin", controller = "Home", action = "Index", id = UrlParameter.Optional },
namespaces: new string[] { "Wenku.Admin.Controllers" },
priority:
); //注册手机访问WebApi路由
context.MapHttpRoute(
name: "Mobile_Api",
routeTemplate: "api/mobile/{controller}/{action}/{id}",
defaults: new
{
area = "mobile",
action = RouteParameter.Optional,
id = RouteParameter.Optional,
namespaceName = new string[] { "Wenku.Mobile.Http" }
},
constraints: new { action = new StartWithConstraint() },
priority:
);
}
}

四、总结

这是一个对Asp.Net Mvc的一个很小的功能拓展,小项目可能不太需要这个功能,但有时候项目大了注册的路由不生效时你应该要想到有可能是因为路由顺序的原因,这时这个路由优先级的功能有可能就会给你带来便利。总之共享给有需要的朋友们参考。

给Asp.Net MVC及WebApi添加路由优先级的更多相关文章

  1. 【转载】为ASP.NET MVC及WebApi添加路由优先级

    路由方面的: 转载地址:http://www.jb51.net/article/73417.htm Author:lijiao 这是一个对Asp.Net Mvc的一个很小的功能拓展,小项目可能不太需要 ...

  2. AJAX跨域调用ASP.NET MVC或者WebAPI服务

    关于AJAX跨域调用ASP.NET MVC或者WebAPI服务的问题及解决方案 作者:陈希章 时间:2014-7-3 问题描述 当跨域(cross domain)调用ASP.NET MVC或者ASP. ...

  3. Asp.Net MVC 进阶篇:路由匹配 实现博客路径 和文章路径

    Asp.Net MVC 进阶篇:路由匹配 实现博客路径 和文章路径 我们要实现 通过路由 匹配出 博客地址 和博客文章地址 例如下面的这两个地址 //http://www.cnblogs.com/ma ...

  4. ASP.NET MVC对WebAPI接口操作(添加,更新和删除)

    昨天<怎样操作WebAPI接口(显示数据)>http://www.cnblogs.com/insus/p/5670401.html 既有使用jQuery,也有使作HttpClient来从数 ...

  5. asp.net MVC5为WebAPI添加命名空间的支持

    前言 默认情况下,微软提供的MVC框架模板中,WebAPI路由是不支持Namespace参数的.这导致一些比较大型的项目,无法把WebApi分离到单独的类库中. 本文将提供解决该问题的方案. 微软官方 ...

  6. AJAX跨域调用ASP.NET MVC或者WebAPI服务的解决方案

    问题描述 当跨域(cross domain)调用ASP.NET MVC或者ASP.NET Web API编写的服务时,会发生无法访问的情况. 重现方式 使用模板创建一个最简单的ASP.NET Web ...

  7. 关于AJAX跨域调用ASP.NET MVC或者WebAPI服务的问题及解决方案

      作者:陈希章 时间:2014-7-3 问题描述 当跨域(cross domain)调用ASP.NET MVC或者ASP.NET Web API编写的服务时,会发生无法访问的情况. 重现方式 使用模 ...

  8. 转:在ASP.NET MVC中通过URL路由实现对多语言的支持

    对于一个需要支持多语言的Web应用,一个很常见的使用方式就是通过请求地址来控制界面呈现所基于的语言文化,比如我们在表示请求地址的URL中将上语言文化代码(比如en或者en-US)来指导服务器应该采用怎 ...

  9. Asp.net MVC 基于Area的路由映射

    对于一个较大规模的Web应用,我们可以从功能上通过Area将其划分为较小的单元.每个Area相当于一个独立的子系统,具有一套包含Models.Views和Controller在内的目录结构和配置文件. ...

随机推荐

  1. HTML基础(五)——-css样式表——样式属性——格式与布局

    一.position:fixed 锁定位置(相对于浏览器的位置),例如有些网站的右下角的弹出窗口. 示例: 二.position:absolute     绝对位置: 1.外层没有position:a ...

  2. Linux 内核日志——dmesg

    有时Linux系统或者系统上运行的mysqld或者其它进程,会发生一些莫名其妙的问题,比如突然挂掉了,比如突然重启等等.在软件上找不到问题所在,此时我们应该怀疑硬件或者内核的问题,此时我们就可以使用 ...

  3. 单片机实现60s定时器

    单片机573+数码管+按钮 实现60秒的定时器 知识: IE寄存器 TCON寄存器 TMOD 寄存器 /***************** 2个定时中断,2个按钮中断 **************** ...

  4. C语言中链表任意位置怎么插入数据?然后写入文件中?

    链表插入示意图:(图是个人所画)因为链表指针指来指去,难以理解,所以辅助画图更加方便. 插入某个学号后面图: 定义的结构体: struct student { ]; //学生学号 ]; //学生姓名 ...

  5. ssh 无密码登录 非相同用户

    场景,机器A 用户a,想登录机器B ,机器B上没有用户a,有用户b. 已知机器B的用户密码,可以这么做. 实验:两台机器都是linux centos的系统. 在机器A上生成a用户的密钥. ssh-ke ...

  6. [译] 企业级 OpenStack 的六大需求(第 1 部分):API 高可用、管理和安全

    全文包括三部分: 第一部分:API 高可用和管理以及安全模型 第二部分:开放架构和混合云兼容 第三部分:弹性架构和全球交付 引言 OpenStack 是构造企业级私有云的非常理想的基础.它立志成为新一 ...

  7. HashMap的工作原理深入再深入

    前言 首先再次强调hashcode (==)和equals的真正含义(我记得以前有人会说,equals是判断对象内容,hashcode是判断是否相等之类): equals:是否同一个对象实例.注意,是 ...

  8. NOIP2009 提高组T3 机器翻译 解题报告-S.B.S

    题目背景 小晨的电脑上安装了一个机器翻译软件,他经常用这个软件来翻译英语文章. 题目描述 这个翻译软件的原理很简单,它只是从头到尾,依次将每个英文单词用对应的中文含义来替换.对于每个英文单词,软件会先 ...

  9. leetcode-Excel Sheet Column Title

    题目: 把数字转化为excel形式的字符表示.示例:1->A 2->B 3->C ... 26->Z 27->AA... 解题思路: 乍一看有点像进制转换题目,不过细想想 ...

  10. CF 407B Long Path[观察性质 DP]

    B. Long Path time limit per test 1 second memory limit per test 256 megabytes input standard input o ...