asp.NET:未知长度的MVC路径

时间:2009-08-26 06:49:43

标签: asp.net-mvc asp.net-mvc-routing

我正在asp.NET中为Web门户构建MVC应用程序。我准备了一系列控制器,并将所有不与之相关的路径映射到一个页面控制器,这将呈现适当的页面。

我的默认路线是这样的:

routes.MapRoute(
  "Default",
  "{level1}/{level2}/{level3}",
  new { controller = "Page", action = "Index", level1 = "home", level2 = "", level3 = "" }
      );

但这有固定的宽度,它最多只能接受3个等级。此外,我想管理附加到路径的操作,例如“编辑”和“删除”。这可能吗?

company/about/who_we_are/staff -> Controller: Page, Action: Index, Parms: company/about/who_we_are/staff
company/about/who_we_are/staff/edit  -> Controller: Page, Action: Edit, Parms: company/about/who_we_are/staff
company/edit  -> Controller: Page, Action: Edit, Parms: company

或者有更好的方法对此进行建模吗?页面的所有路径都在数据库中,因此它们会动态更改。

3 个答案:

答案 0 :(得分:5)

您可以使用外卡路线:

"{*data}"

看看这个SO:ASP.net MVC custom route handler/constraint


简单可能的解决方案:

(未经测试,但......)

路线:

routes.Add(new Route
                           (
                           "{*data}",
                           new RouteValueDictionary(new {controller = "Page", action = "Index", data = ""}),
                           new PageRouteHandler()
                           )
                );

处理程序看起来像:

public class PageRouteHandler : IRouteHandler
{
    public IHttpHandler GetHttpHandler(RequestContext requestContext)
    {
        return new PageHttpHandler(requestContext);
    }
}

class PageHttpHandler : MvcHandler
{
    public PageHttpHandler(RequestContext requestContext)
        : base(requestContext)
    {
    }

    protected override void ProcessRequest(HttpContextBase httpContext)
    {
        IController controller = new PageController();

        ((Controller)controller).ActionInvoker = new PageActionInvoker();

        controller.Execute(RequestContext);
    }
}

class PageActionInvoker : ControllerActionInvoker
{
    protected override ActionResult InvokeActionMethod(ControllerContext controllerContext, ActionDescriptor actionDescriptor, IDictionary<string, object> parameters)
    {
        string data = controllerContext.RouteData.GetRequiredString("data");
        string[] tokens = data.Split('/');


        int lenght = tokens.Length;

        if (lenght == 0)                   
            return new NotFoundResult();

        if (tokens[tokens.Length - 1] == "edit")
        {
            parameters["action"] = "edit";
            lenght--;
        }

        for (int i = 0; i < length; i++)
            parameters["level" + (i + 1).ToString()] = tokens[i];

        return base.InvokeActionMethod(controllerContext, actionDescriptor, parameters);
    }
}

答案 1 :(得分:2)

网址中任何地方的贪婪段可能吗?是的!

我写过GreedyRoute类,它支持URL中任何位置的贪婪(catch all)段。你需要它已经有一段时间了,但是将来可能对其他人有用。

它支持以下任何模式:

  • {segment}/{segment}/{*greedy} - 默认Route
  • 已支持此功能
  • {segment}/{*greedy}/{segment} - 中间贪婪
  • {*greedy}/{segment}/{segment} - 贪婪

您可以阅读所有详细信息on my blog post并获取代码。

答案 2 :(得分:1)

据我所知,您可以使用正则表达式来表示路径的外观(请参阅底部代码部分here)。有了这个,应该可以创建一个可以采用不确定数量的子部分的正则表达式字符串(“forward-slashe和text / number-groups”)。然后,您可以解析应用程序中的URL字符串并检索相应的部分。

然而,我不能在不花费数小时的情况下自己编写这个正则表达式字符串,所以其他人可能会帮助你。 : - )