【问题标题】:UrlHelper.Action includes undesired additional parametersUrlHelper.Action 包含不需要的附加参数
【发布时间】:2013-12-03 11:18:12
【问题描述】:

我在控制器 ApplicationsController 中有一个方法,我需要在其中获取操作方法的基本 URL:

public ActionResult MyAction(string id)
{
    var url = Url.Action("MyAction", "Applications");
    ...
}

问题是这包括来自当前路由数据的string id,当我需要没有 URL 时(该 URL 用于在基于 URL 的查找中从 CMS 获取内容)。

我尝试将nullnew { } 作为routeValues 参数传递,但无济于事。

匹配的路由如下(高于所有其他路由):

routes.MapLowercaseRoute(
    name: "Applications",
    url: "applications/{action}/{id}",
    defaults: new { controller = "Applications",
                    action = "Index", id = UrlParameter.Optional });

我已经看到其他几个问题涉及此问题,但似乎都没有可行的解决方案。目前,我正在诉诸于对控制器中的路径进行硬编码;但是,我希望能够将其抽象为动作过滤器,因此我需要能够生成 URL。

是否有一种干净/传统的方法来防止这种行为?

【问题讨论】:

  • 你确定这条路线解决了吗? (如果它是唯一的)。
  • @ThomasJaskula - 肯定的 - 我在它下面有一条包罗万象的路线,它直接重定向到静态 CMS 内容。评论应用程序路由会导致第二条路由捕获它。已将其作为第一个停靠港进行测试,这是唯一可以解析该 URL 的路由。

标签: c# asp.net-mvc-4 urlhelper


【解决方案1】:

好的,我看到了问题。这就是所谓的“段变量重用”。在为出站 URL 生成路由并尝试在路由的 URL 模式中查找每个分段变量的值时,路由系统将查看当前请求中的值。这种行为使许多程序员感到困惑,并可能导致冗长的调试会话。路由系统热衷于对路由进行匹配,以至于它将重用来自传入 URL 的段变量值。所以我认为你必须像 Julien 建议的那样覆盖这个值:

var url = Url.Action("MyAction", "Applications", new { id = "" })

【讨论】:

  • 有趣。会试一试,但是有没有办法在不明确指定各个参数名称的情况下强制这种情况发生?我想将其抽象为一个动作过滤器,其中 URL 段不一定总是被称为 id
  • 我不认为有一种简单的方法可以做到这一点,因为它已经融入了 MVC 框架。它需要定制路由系统。这种行为总是让我大吃一惊,这就是为什么我认为路由是 MVC 中最糟糕的部分。这不应该经常发生,我正在逐案处理。
  • 我在使用 UrlHelper.Link 时遇到了这种情况,但这并不能解决问题
【解决方案2】:

最终用不同的方法解决了这个问题。为了防止将任意命名的路由值插入到生成的 URL 中,我能想到的唯一方法是在调用 Url.Action 时将它们从 RouteData 中临时删除。我写了几个扩展方法来促进这一点:

public static string NonContextualAction(this UrlHelper helper, string action)
{
    return helper.NonContextualAction(action,
        helper.RequestContext.RouteData.Values["controller"].ToString());
}

public static string NonContextualAction(this UrlHelper helper, string action,
                                         string controller)
{
    var routeValues = helper.RequestContext.RouteData.Values;
    var routeValueKeys = routeValues.Keys.Where(o => o != "controller"
                         && o != "action").ToList();

    // Temporarily remove routevalues
    var oldRouteValues = new Dictionary<string, object>();
    foreach (var key in routeValueKeys)
    {
        oldRouteValues[key] = routeValues[key];
        routeValues.Remove(key);
    }

    // Generate URL
    string url = helper.Action(routeValues["Action"].ToString(),
                               routeValues["Controller"].ToString());

    // Reinsert routevalues
    foreach (var kvp in oldRouteValues)
    {
        routeValues.Add(kvp.Key, kvp.Value);
    }

    return url;
}

这允许我在操作过滤器中执行此操作,我不一定知道操作的参数名称是什么(因此不能像其他答案那样只传递匿名对象)。

但是,仍然非常想知道是否有人有更优雅的解决方案。

【讨论】:

  • 非常好的方法。事实上,Url 类本身应该具有这些行为的替代方案(例如 Action 方法上的可选 bool 参数)。
  • @MarceloMyara 这也是我的想法——事实上,我很惊讶没有考虑到这一点。
【解决方案3】:

id 使用null 或空值以防止Url.Action 使用当前值:

var url = Url.Action("MyAction", "Applications", new { id = "" })

【讨论】:

    【解决方案4】:

    在@AntP 的其他很好的解决方案中,我对 RouterData 的更改、暂时性或其他方式并不完全满意。由于创建链接的代码已经集中化,我借用了@Tomasz Jaskuλa 和@AntP 来扩充我已经在使用的ExpandoObject

    IDictionary<string,object> p = new ExpandoObject();
    
    // Add the values I want in the route
    foreach (var (key, value) in linkAttribute.ParamMap)
    {
        var v = GetPropertyValue(origin, value);                    
        p.Add(key, v); 
    }
    
    // Ideas borrowed from https://stackoverflow.com/questions/20349681/urlhelper-action-includes-undesired-additional-parameters
    // Null out values that I don't want, but are already in the RouteData
    foreach (var key in _urlHelper.ActionContext.RouteData.Values.Keys)
    {
        if (p.ContainsKey(key))
            continue;
    
        p.Add(key, null); 
    }
    
    var href = _urlHelper.Action("Get", linkAttribute.HRefControllerName, p);
    

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 2015-04-02
      • 1970-01-01
      • 2021-03-28
      • 2011-08-25
      • 2011-09-13
      • 1970-01-01
      • 2015-02-01
      • 2012-03-04
      相关资源
      最近更新 更多