MVC路由中的控制器类别? (在单独的命名空间中重复的控制器名称)

时间:2020-03-05 18:47:28  来源:igfitidea点击:

我正在寻找以下情况的一些路由示例或者示例:

做事的一般示例是:{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            
        });
    }

唯一的另一件事是它将引用仍位于基本目录中的视图,因此,如果将视图放入匹配的目录中,则在将其返回到控制器内部时必须将视图名称放入其中。