Маршрутизация для действий с одинаковыми именами, но с разными параметрами
У меня есть этот набор маршрутов:
routes.MapRoute(
"IssueType",
"issue/{type}",
new { controller = "Issue", action = "Index" }
);
routes.MapRoute(
"Default", // Route name
"{controller}/{action}/{id}", // URL with parameters
new { controller = "Home", action = "Index", id = UrlParameter.Optional } // Parameter defaults
);
Вот класс контроллера:
public class IssueController : Controller
{
public ActionResult Index()
{
// todo: redirect to concrete type
return View();
}
public ActionResult Index(string type)
{
return View();
}
}
почему, когда я запрашиваю http://host/issue, я получаю The current request for action 'Index' on controller type 'IssueController' is ambiguous between the following action methods:
Я ожидаю, что первый метод должен действовать, когда нет параметров, а второй - при заданном параметре.
где я совершил ошибку?
UPD: возможный дубликат: Можете ли вы перегрузить методы контроллера в ASP.NET MVC?
UPD 2: из-за ссылки выше - нет никакого законного способа сделать перегрузку действий, не так ли?
UPD 3: методы действий не могут быть перегружены на основе параметров (c) http://msdn.microsoft.com/en-us/library/system.web.mvc.controller%28VS.100%29.aspx
Ответы
Ответ 1
У меня бы был один метод Index, который ищет допустимую переменную типа
public class IssueController : Controller
{
public ActionResult Index(string type)
{
if(string.isNullOrEmpty(type)){
return View("viewWithOutType");}
else{
return View("viewWithType");}
}
}
EDIT:
Как создать пользовательский атрибут, который ищет конкретное значение запроса, как в этом сообщении fooobar.com/questions/50643/...
[RequireRequestValue("someInt")]
public ActionResult MyMethod(int someInt) { /* ... */ }
[RequireRequestValue("someString")]
public ActionResult MyMethod(string someString) { /* ... */ }
public class RequireRequestValueAttribute : ActionMethodSelectorAttribute {
public RequireRequestValueAttribute(string valueName) {
ValueName = valueName;
}
public override bool IsValidForRequest(ControllerContext controllerContext, MethodInfo methodInfo) {
return (controllerContext.HttpContext.Request[ValueName] != null);
}
public string ValueName { get; private set; }
}
Ответ 2
Я столкнулся с аналогичной ситуацией, когда мне нужно, чтобы мое действие "Индекс" обрабатывало рендеринг, если у меня был ID или нет. Решение, с которым я столкнулся, состояло в том, чтобы сделать параметр ID опционным методом Index.
Например, я изначально пытался иметь оба:
public ViewResult Index()
{
//...
}
// AND
public ViewResult Index(int entryId)
{
//...
}
и я просто объединил их и изменил на:
public ViewResult Index(int entryId = 0)
{
//...
}
Ответ 3
Вы можете сделать это, используя ActionFilterAttribute, который проверяет параметры с помощью отражения (я попробовал), но это плохая идея. Каждое отдельное действие должно иметь свое собственное имя.
Почему бы просто не назвать ваши два метода "Индекс" и "Одиночный", скажем, и жить с ограничением на именование?
В отличие от методов, которые привязаны во время компиляции на основе совпадающих подписей, значение отсутствующего маршрута в конце обрабатывается как нуль.
Если вы хотите, чтобы [hack] ActionFilterAttribute соответствовал параметрам, дайте мне знать, и я отправлю ему ссылку, но, как я уже сказал, это плохая идея.
Ответ 4
Все, что вам нужно сделать, это отметить свое второе действие с помощью [HttpPost]. Например:
public class IssueController : Controller
{
public ActionResult Index()
{
// todo: redirect to concrete type
return View();
}
[HttpPost]
public ActionResult Index(string type)
{
return View();
}
}