13

我正在处理的网站有一些相当复杂的路由结构,我们在使用路由引擎以我们需要的方式构建 URL 时遇到了一些困难。

我们有一个搜索结果页面,它使用基于 RegEx 的模式匹配将多个变量分组到单个路由段中(即“www.host.com/{structuralParameters}”可以如下:“www.host.com/variableA-variableB- variableC" - 其中变量 A 到 C 都是可选的)。经过一些工作,这对我们来说很好。

我们遇到的问题解决了 ActionLink 方法的一个烦人的特性:如果你指向同一个控制器/动作,无论你是否想要它们,它都会保留现有的路由值。我们更愿意控制链接的外观,并且在某些情况下,不能保留现有参数。一个例子是我们网站的主导航指向一个没有设置参数的搜索结果页面 - 如果您愿意,可以使用默认搜索页面。我说这是一个令人讨厌的功能,因为它是 ASP.Net MVC 框架的一个罕见实例,似乎在没有明显扩展点的情况下指示实现——我们宁愿不创建自定义 ActionLink 代码来在我们的母版页中编写简单的导航链接!

我看到有人说您需要将此类参数显式设置为空字符串,但是当我们尝试这样做时,它只是将参数从路由值更改为查询字符串参数。在我看来,我们应该被要求明确排除我们没有作为参数明确传递给 ActionLink 方法的值,但如果这是我们唯一的选择,我们将使用它。但是目前如果它显示在查询字符串中,那么它对我们来说就像直接将参数放入路由中一样无用。

我知道我们的路由结构加剧了这个问题 - 如果我们使用更简单的方法(即 www.host.com/variableA/variableB/variableC),我们可能不会有任何问题,但我们的 URL 结构是不可协商的 - 它是旨在满足与可用性、搜索引擎优化和链接/内容共享相关的非常具体的需求。

我们如何使用 Html.ActionLink 生成页面链接而不依赖于当前的路由数据(或者,如果可能,需要显式排除路由段),即使这些链接导致相同的操作方法?

如果我们确实需要明确排除路由段,我们如何防止该方法将路由呈现为查询字符串参数?

这个看似很小的问题给我们带来了令人惊讶的悲伤,我将感谢任何帮助解决它。

编辑:按照 LukLed 的要求,这是一个 ActionLink 调用示例:

// I've made it generic, but this should call the Search action of the 
// ItemController, the text and title attribute should say "Link Text" but there
// should be no parameters - or maybe just the defaults, depending on the route.
// 
// Assume that this can be called from *any* page but should not be influenced by
// the current route - some routes will be called from other sections with the same
// structure/parameters.
Html.ActionLink( 
    "Link Text",
    "Search", 
    "Item", 
    new { }, 
    new { title = "Link Text" } 
);
4

3 回答 3

21

Html.ActionLink在调用or (或任何 URL 生成方法)时将路由值设置为 null 或空字符串Html.RouteLink将清除“环境”路由值。

例如,使用标准 MVC 控制器/动作/id 路由,假设您在“Home/Index/123”上。如果您调用,则 MVC 将注意到andHtml.RouteLink(new { id = 456 })的“环境”路由值。它还会注意到环境路由值,但它会被显式的“456”覆盖。这将导致生成的 URL 为“Home/Index/456”。controller="Home"action="Index"id="123"

参数的顺序也很重要。例如,假设您打电话给Html.RouteLink(new { action = "About" }). “关于”动作将覆盖当前的“索引”动作,而“id”参数将被完全清除!但你问为什么?因为一旦你使一个参数段失效,那么它之后的所有参数段都将失效。在这种情况下,“action”被一个新的显式值无效,因此紧随其后且没有显式值的“id”也被无效。因此,生成的 URL 将只是“Home/About”(没有 ID)。

在同样的情况下,如果您调用Html.RouteLink(new { action = "" }),则生成的 URL 将只是“Home”,因为您使用空字符串使“action”无效,然后导致“id”也无效,因为它出现在无效的“action”之后”。

于 2009-12-01T18:33:36.807 回答
4

解决问题的根源

似乎最佳解决方案(闻起来不像是一种解决方法)是解决问题的根源并且在路由中的解决方案。

我编写了一个Route名为的自定义类RouteWithExclusions,它能够定义在生成 URL 时应该排除/删除的路由值名称。问题是当路由通过路由表并且后续路由没有相同的路由值名称时......

整个问题在我的博客文章中进行了详细说明和解释,所有代码也都在那里提供。检查一下,它可能会帮助您解决此路由问题。我还编写了两个额外MapRoute的扩展方法,它们带有一个额外的参数。

于 2011-03-17T14:29:17.393 回答
1

如果您想完全控制链接,只需自己构建链接:

<a href="~/variableA/variableB/<%= Html.Encode(Model.Target) %>">Click Here</a>

href属性中替换您需要的任何内容。

于 2009-11-30T02:34:14.933 回答