有时候我们会碰到两个项目合在一起,那么必然会碰到两个同名的controller,其实MVC在注册路由,添加Route的时候可以指定当前规则解析那个命名空间下的所有Controller。
注:Controller的调用是通过IControllerFactory,反射调用目标Controller,不指定目标命名空间,直接从BuildManager.GetReferencedAssemblies();如下。
private static List<Type> controllerTypes = new List<Type>(); /// <summary> /// 静态构造函数 /// </summary> static DefaultControllerFactory() { var assemblys = BuildManager.GetReferencedAssemblies(); foreach (Assembly assembly in assemblys) { var types = assembly.GetTypes().Where(a => typeof(IController).IsAssignableFrom(a)); foreach (Type type in types) { controllerTypes.Add(type); } } } /// <summary> /// 创建controller /// </summary> /// <param name="requestContext"></param> /// <param name="controllerName"></param> /// <returns></returns> public IController CreateController(RequestContext requestContext, string controllerName) { string typeName = controllerName + "Controller"; Type controllerType = controllerTypes.FirstOrDefault(a => a.Name == typeName); if (controllerType != null) { return (IController)Activator.CreateInstance(controllerType); } return null; }
回归正题:如何支持多种命名空间
public static void RegisterRoutes(RouteCollection routes) { routes.IgnoreRoute("{resource}.axd/{*pathInfo}"); routes.MapRoute( name: "Default", url: "{controller}/{action}", defaults: new { controller = "Home", action = "Index" }, namespaces: new string[] { "MvcExpose.Controllers" } ); routes.MapRoute("Admin", "Admin/{controller}/{action}", new { controller = "Home", action = "Index"}, new string[] { "MvcExpose.Admin.Controllers" }); }
切记:两种注册Url规则,要长度不一致,Default带有"{controller}/{action}/{id}"一直报错,以为在正则解析看来,无法区分两种路由规则的区别,因此当“http://localhost:4500/Admin/Home/Index”,系统先调用default的路由解析。
其实有一种更好的的方式:
就是建立所谓的区域,每一个区域就是一个独立的子系统,如下图:
其实最为重要的是用了AreaRegistration
public class AdminAreaRegistration : AreaRegistration { public override string AreaName { get { return "Admin"; } } public override void RegisterArea(AreaRegistrationContext context) { context.MapRoute( "Admin_default", "Admin/{controller}/{action}/{id}", new { controller="home",action = "Index", id = UrlParameter.Optional } ); } }
在Global中,我们有一句: AreaRegistration.RegisterAllAreas();调用这个方法的时候,当前Web应用所有直接或间接被引用的程序集会被加载,然后从这些程序集中解析出所有继承自AreaRegistration的类型并反射出对象,调用相应的RegisterArea。
protected void Application_Start() { AreaRegistration.RegisterAllAreas(); WebApiConfig.Register(GlobalConfiguration.Configuration); FilterConfig.RegisterGlobalFilters(GlobalFilters.Filters); RouteConfig.RegisterRoutes(RouteTable.Routes); }
原理:
MVC调用Controller是通过反射程序集中继承了IController的所有类(默认情况下),根据路由规则取出ControllerName,并实例化相对应的Controller实例,如果出现重名的Controller,会报错。
因此要指定好对应的命名空间,MVC将namespace存在RouteData的DateTokens中
十分直观的的看到,这样就给不同的路由规则,反射相应的命名空间下IController的实例!