MVC路由中的控制器类别? (在单独的命名空间中重复的控制器名称)
我正在寻找以下情况的一些路由示例或者示例:
做事的一般示例是:{controller} / {action} / {id}
因此,在进行产品搜索商店的情况下,我们将拥有:
public class ProductsController: Controller { public ActionResult Search(string id) // id being the search string { ... } }
假设我们有几家商店,并且始终如一地希望这样做,那么有什么办法可以做到:{category} / {controller} / {action} / {id}
这样我们可以对特定商店进行特定搜索,但对不同商店使用不同的搜索方法?
(如果我们要求商店名称的优先级高于URL中的函数本身)
还是会归结为:
public class ProductsController: Controller { public ActionResult Search(int category, string id) // id being the search string { if(category == 1) return Category1Search(); if(category == 2) return Category2Search(); ... } }
这可能不是一个很好的例子,但基本上的想法是使用相同的控制器名称,因此在几种不同的情况下都有一个简单的URL,或者我们是否坚持要求唯一的控制器名称,而没有办法将它们放入略有不同的名称空间/目录?
编辑添加:
我想要这样做的另一个原因是,我可能想要具有类别的URL,并且某些控制器只能在某些类别下工作。
IE:
/ this / search / items / search + term <-作品
/ that / search / items / search + term <-无效,因为不允许使用搜索控制器。
解决方案
回答
做到这一点的最好方法是,通过继承IControllerFactory来实现自己的ControllerFactory。我们将实现的CreateController方法将处理创建控制器实例以处理RouteHandler和ControllerActionInvoker的请求。约定是在创建控制器时使用控制器的名称,因此我们将需要覆盖此功能。在这里,我们将放置用于根据路由创建控制器的自定义逻辑,因为我们将有多个具有相同名称但在不同文件夹中的控制器。然后,我们将需要在应用程序启动时注册自定义控制器工厂,就像路线一样。
我们需要考虑的另一个领域是在创建控制器时查找视图。如果我们计划对所有视图使用相同的视图,那么我们不必做任何与所使用的约定不同的事情。如果我们还计划组织视图,则还需要创建自己的ViewLocator,并在控制器工厂中创建它时将其分配给控制器。
为了了解代码,我在SO上回答了几个与此问题相关的问题,但这在某种程度上有所不同,因为控制器名称相同。我提供了一些参考链接。
- ASP.NET MVC中单独程序集中的视图
- asp.net mvc-子文件夹
另一条路线,但是可能需要做出一些妥协,将是使用新的AcceptVerbs属性。请查看此问题以获取更多详细信息。我还没有玩过这个新功能,但这可能是另一条路。
回答
实际上,我什至没有通过搜索找到它,而是通过在这个问题中浏览ASP.NET论坛发现的。
使用此方法,我们可以在名称空间的任何部分下使用相同名称的控制器,只要我们限定哪些路由属于哪个名称空间即可(如果需要,每个路由可以有多个名称空间!)
但是从这里,我们可以在控制器下放置一个目录,因此,如果控制器是" MyWebShop.Controllers",则将放置目录" Shop1",而名称空间将是" MyWebShop.Controllers.Shop1"
然后工作:
public static void RegisterRoutes(RouteCollection routes) { routes.IgnoreRoute("{resource}.axd/{*pathInfo}"); var shop1namespace = new RouteValueDictionary(); shop1namespace.Add("namespaces", new HashSet<string>(new string[] { "MyWebShop.Controllers.Shop1" })); routes.Add("Shop1", new Route("Shop1/{controller}/{action}/{id}", new MvcRouteHandler()) { Defaults = new RouteValueDictionary(new { action = "Index", id = (string)null }), DataTokens = shop1namespace }); var shop2namespace = new RouteValueDictionary(); shop2namespace.Add("namespaces", new HashSet<string>(new string[] { "MyWebShop.Controllers.Shop2" })); routes.Add("Shop2", new Route("Shop2/{controller}/{action}/{id}", new MvcRouteHandler()) { Defaults = new RouteValueDictionary(new { action = "Index", id = (string)null }), DataTokens = shop2namespace }); var defaultnamespace = new RouteValueDictionary(); defaultnamespace.Add("namespaces", new HashSet<string>(new string[] { "MyWebShop.Controllers" })); routes.Add("Default", new Route("{controller}/{action}/{id}", new MvcRouteHandler()) { Defaults = new RouteValueDictionary(new { controller = "Home", action = "Index", id = "" }), DataTokens = defaultnamespace }); }
唯一的另一件事是它将引用仍位于基本目录中的视图,因此,如果将视图放入匹配的目录中,则在将其返回到控制器内部时必须将视图名称放入其中。