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)
尽管您的问题尚不清楚,但从其他答案的评论来看,您似乎无法控制向您发送此 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 文章,可能会让您更深入地了解绑定在幕后的工作原理。
| 归档时间: |
|
| 查看次数: |
1704 次 |
| 最近记录: |