如何使用相同的 MVC 控制器方法处理数组或单个对象?

4th*_*ace 5 c# json asp.net-mvc-4

我有一个 ActionResult 方法,它接受列表作为参数:

[HttpPost]
public ActionResult MyMethod (List<ClassA> json)
{
   ...
}
Run Code Online (Sandbox Code Playgroud)

这会将传入的 json 字符串绑定到已填充的 ClassA 对象的通用列表。

问题是有时传入的 json 只是一个 json 对象,而不是 json 对象数组。

有什么方法可以抢占这个,这样我就可以直接绑定到 ClassA 与 List 吗?或者我可以使用其他一些技术吗?

以下是 JSON 的发送方式(作为数组):

var myjsonarray = [{
  "ID": "1",
  "FirstName": "John",
  "LastName": "Doe",
}, {
  "ID": "2",
  "FirstName": "Jane",
  "LastName": "Doe",
}];

$.ajax({
 type: "POST",
 contentType: "application/json; charset=utf-8",
 url: "/Home/MyPage",
 data: JSON.stringify(myjsonarray),
 dataType: 'json'
});
Run Code Online (Sandbox Code Playgroud)

以上处理没问题。这也有效:

var myjsonarray = [{
  "ID": "1",
  "FirstName": "John",
  "LastName": "Doe",
}];
Run Code Online (Sandbox Code Playgroud)

但是当我作为未包装在数组中的单个对象发送时:

var myjsonarray = {
  "ID": "1",
  "FirstName": "John",
  "LastName": "Doe",
};
Run Code Online (Sandbox Code Playgroud)

我的 ActionResult 方法参数为 null:

json == null
Run Code Online (Sandbox Code Playgroud)

Bri*_*ers 5

尽管您的问题尚不清楚,但从其他答案的评论来看,您似乎无法控制向您发送此 JSON 的客户端代码。(如果您这样做,最简单的解决方案是在发送单个对象之前将其包装在数组中,正如其他人已经建议的那样。)

如果 MVC 允许我们在控制器中添加方法重载来处理这种情况,那就太好了,例如:

[HttpPost]
public ActionResult MyMethod (List<ClassA> list)
{
   ...
}

[HttpPost]
public ActionResult MyMethod (ClassA single)
{
   ...
}
Run Code Online (Sandbox Code Playgroud)

虽然编译得很好,但不幸的是,System.Reflection.AmbiguousMatchException当您在运行时点击该方法时,它会导致 。

因此,看起来您需要创建一个自定义IModelBinder来解决该问题。虽然我之前没有实现过自定义模型绑定器,但我将其视为一个挑战并提出了以下建议。显然,您必须根据自己的需要调整它,但它似乎适用于您的示例。

这是代码:

public class CustomModelBinder : IModelBinder
{
    public object BindModel(ControllerContext controllerContext, ModelBindingContext bindingContext)
    {
        IValueProvider provider = bindingContext.ValueProvider;

        // Check whether we have a list or a single object.  If we have a list
        // then all the properties will be prefixed with an index in square brackets.
        if (provider.ContainsPrefix("[0]"))
        {
            // We have a list.  Since that is what the controller method is 
            // expecting, just use the default model binder to do the work for us.
            return ModelBinders.Binders.DefaultBinder.BindModel(controllerContext, bindingContext);
        }
        else
        {
            // We have a single object.
            // Bind it manually and return it in a list.
            ClassA a = new ClassA();
            a.ID = GetValue<int>(provider, "ID");
            a.FirstName = GetValue<string>(provider, "FirstName");
            a.LastName = GetValue<string>(provider, "LastName");
            return new List<ClassA> { a };
        }
    }

    private T GetValue<T>(IValueProvider provider, string key)
    {
        ValueProviderResult result = provider.GetValue(key);
        return (result != null ? (T)result.ConvertTo(typeof(T)) : default(T));
    }
}
Run Code Online (Sandbox Code Playgroud)

要将此自定义绑定器插入到 MVC 管道中,请将此行添加Application_Start()Global.asax.cs.

ModelBinders.Binders.Add(typeof(List<ClassA>), new CustomModelBinder());
Run Code Online (Sandbox Code Playgroud)

现在,显然这不是一个通用的解决方案;它是专门为处理ClassA而定制的,仅此而已。我必须相信可以制定一个通用的解决方案来处理任何类型列表的“单一或列表”情况,但这要复杂得多并且超出了本答案的范围。为此,您可能还需要创建一个自定义IModelBinderProvider. 如果你想达到那个水平,你需要自己进行调查。这是一篇MSDN 文章,可能会让您更深入地了解绑定在幕后的工作原理。