【发布时间】:2010-11-05 22:18:39
【问题描述】:
我有两个相互冲突的操作方法。基本上,我希望能够使用两条不同的路线到达同一个视图,或者通过项目的 ID 或通过项目的名称及其父项的名称(项目可以在不同的父项中具有相同的名称)。可以使用搜索词来过滤列表。
例如...
Items/{action}/ParentName/ItemName
Items/{action}/1234-4321-1234-4321
这是我的操作方法(还有Remove操作方法)...
// Method #1
public ActionResult Assign(string parentName, string itemName) {
// Logic to retrieve item's ID here...
string itemId = ...;
return RedirectToAction("Assign", "Items", new { itemId });
}
// Method #2
public ActionResult Assign(string itemId, string searchTerm, int? page) { ... }
这里是路线...
routes.MapRoute("AssignRemove",
"Items/{action}/{itemId}",
new { controller = "Items" }
);
routes.MapRoute("AssignRemovePretty",
"Items/{action}/{parentName}/{itemName}",
new { controller = "Items" }
);
我明白为什么会发生错误,因为page 参数可以为空,但我想不出解决它的最佳方法。我的设计一开始就很差吗?我考虑过扩展Method #1 的签名以包含搜索参数,并将Method #2 中的逻辑移到他们都会调用的私有方法中,但我认为这不会真正解决歧义。
任何帮助将不胜感激。
实际解决方案(基于李维斯的回答)
我添加了以下类...
public class RequireRouteValuesAttribute : ActionMethodSelectorAttribute {
public RequireRouteValuesAttribute(string[] valueNames) {
ValueNames = valueNames;
}
public override bool IsValidForRequest(ControllerContext controllerContext, MethodInfo methodInfo) {
bool contains = false;
foreach (var value in ValueNames) {
contains = controllerContext.RequestContext.RouteData.Values.ContainsKey(value);
if (!contains) break;
}
return contains;
}
public string[] ValueNames { get; private set; }
}
然后修饰动作方法...
[RequireRouteValues(new[] { "parentName", "itemName" })]
public ActionResult Assign(string parentName, string itemName) { ... }
[RequireRouteValues(new[] { "itemId" })]
public ActionResult Assign(string itemId) { ... }
【问题讨论】:
-
太棒了!小改动建议:(imo 真的很有用)1)params string[] valueNames 使属性声明更简洁(偏好)2)用
return ValueNames.All(v => controllerContext.RequestContext.RouteData.Values.ContainsKey(v));替换 IsValidForRequest 方法体 -
嗨,乔恩,我想我不明白,因为 RouteData 中的查询参数在哪里?
-
我遇到了同样的查询字符串参数问题。如果您需要根据要求考虑的这些参数,请将
contains = ...部分换成如下内容:contains = controllerContext.RequestContext.RouteData.Values.ContainsKey(value) || controllerContext.RequestContext.HttpContext.Request.Params.AllKeys.Contains(value); -
警告说明:所需的参数必须完全按名称发送。如果您的操作方法参数是通过按名称传递其属性来填充的复杂类型(并让 MVC 将它们按摩到复杂类型中),则此系统将失败,因为该名称不在查询字符串键中。例如,这将不起作用:
ActionResult DoSomething(Person p),其中Person具有各种简单的属性,如Name,并且直接使用属性名称(例如,/dosomething/?name=joe+someone&other=properties)向它发出请求。 -
如果你使用 MVC4 以后的版本,你应该使用
controllerContext.HttpContext.Request[value] != null而不是controllerContext.RequestContext.RouteData.Values.ContainsKey(value);不过还是不错的作品。
标签: c# asp.net-mvc asp.net-mvc-routing