4

我有一个超级简单的控制器,有两种方法:

public IActionResult Users(long id)
{
    return Json(new { name = "Example User" });
}

public IActionResult Users()
{
    return Json(new { list = new List<User>() });
}

一个选择所有用户,另一个返回所有用户。在 web api 2 中,我可以使用以下路线并且一切正常:

config.Routes.MapHttpRoute(
                name: "Users",
                routeTemplate: "v1/Users",
                defaults: new { action = "Users", controller = "Users" },
                constraints: null,
                handler: new TokenValidationHandler() { InnerHandler = new HttpControllerDispatcher(config) }
            );

我在 startup.cs 中设置了以下路由:

app.UseMvc(routes =>
            {
                routes.MapRoute(name: "User_Default", template: "v1/{controller=Users}/{action=Users}/{id?}");
            });

然而,这给了我一个AmbiguousActionException: Multiple actions matched. The following actions matched route data and had all constraints satisfied

我究竟做错了什么?

4

1 回答 1

6

在您的原始 webapi 代码中,您使用Routes.MapHttpRoute的是添加 webapi 特定路由。这与不考虑动作参数的 MVC 路由不同,例如,如果您使用Routes.MapRoute.

同样的事情也发生在您的 MVC 6 代码中,因为您正在使用routes.MapRoute. 在这两种情况下,框架都会找到 2 个与相同路由匹配且没有额外约束的控制器操作。它需要一些帮助才能选择这两个动作之一。

消除 api 操作歧义的最简单方法是使用属性路由而不是定义路由,如下例所示

[Route("v1/[controller]")]
public class UsersController : Controller
{
    [HttpGet("{id:int}")]
    public IActionResult Users(long id)
    {
        return Json(new { name = "Example User" });
    }

    public IActionResult Users()
    {
        return Json(new { list = new[] { "a", "b" } });
    }
}

还有其他选项可以让您更改 MVC 6 中 MVC 路由的行为。您可以创建自己的IActionConstraint属性来强制具有或不具有给定参数。这样,其中一个操作需要路由中的 id 参数,而另一个不需要 id 参数(警告,未经测试的代码):

public class UsersController : Controller
{
    [RouteParameterConstraint("id", ShouldAppear=true)]
    public IActionResult Users(long id)
    {
        return Json(new { name = "Example User" });
    }

    [RouteParameterConstraint("id", ShouldNotAppear=true)]
    public IActionResult Users()
    {
        return Json(new { list = new[] { "a", "b" } });
    }
}

public class RouteParameterConstraintAttribute : Attribute, IActionConstraint
{
    private routeParameterName;

    public RouteParameterConstraintAttribute(string routeParameterName)
    {
        this.routerParamterName = routerParameterName;
    }

    public int Order => 0;
    public bool ShouldAppear {get; set;}
    public bool ShouldNotAppear {get; set;}

    public bool Accept(ActionConstraintContext context)
    {
        if(ShouldAppear) return context.RouteContext.RouteData.Values["country"] != null;
        if(ShouldNotAppear) return context.RouteContext.RouteData.Values["country"] == null;

        return true;
    }
}

处理 webapi 2 样式控制器的更好选择是将约定添加到 MVC 管道中。这正是Microsoft.AspNet.Mvc.WebApiCompatShim帮助迁移 webapi 2 控制器所做的事情。您可以在此处查看添加的约定。查看本指南以快速了解此软件包。

于 2015-06-21T13:19:30.550 回答