Подтвердить что ты не робот

Маршрутизация для действий с одинаковыми именами, но с разными параметрами

У меня есть этот набор маршрутов:

        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

4b9b3361

Ответ 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();
    }
}