Json пустой десериализации массива как null в MVC
У меня есть действие контроллера, которое должно получать целое число и объект, содержащий различные свойства, один из которых является общим списком объектов. Когда я отправляю JSON в действие с заполненным списком, все правильно отображает и я получаю список, содержащий объект, который я разместил. Если массив пуст, действие MVC привязывает свойство к нулевому списку пустого списка. Я хочу, чтобы пустой массив отображался в пустой массив, а не в нуль, поскольку пустой массив в этом случае означает, что в коллекции нет ничего, а нуль означает, что база данных должна быть проверена, чтобы увидеть, есть ли что-либо ранее сохранен в коллекции, но я не могу понять, что мне нужно изменить, чтобы правильно отобразить его. Мы используем Json.Net для сериализации объектов для возвращаемых объектов, но я не думаю, что он используется для десериализации объектов при привязке модели.
Передаваемые объекты:
public class ObjectInList
{
public decimal Value1 { get; set; }
public decimal Value2 { get; set; }
}
public class Criteria
{
public decimal? ANullableNumber { get; set; }
public IList<ObjectInList> ObjectsList { get; set; }
}
Запрос Json:
"{\" Идентификатор \ ": 137, \" критерии \ ": {\" ObjectsList\ ": []}}"
Действие контроллера:
public ActionResult ProcessCriteria(int id, Criteria criteria)
{
return Json(_service.ProcessCriteria(id, criteria));
}
В действии контроллера я получаю нуль вместо пустого списка в объекте критериев. Это происходит, если я посылаю nulls для других свойств или нет. Не уверен, что до объекта, являющегося IList, а не IEnumerable? (Метод Json, обертывающий вызов службы, является нашей оболочкой, чтобы вернуть результат json, используя Json.Net для сериализации ответа - нуль находится в объекте критериев, а не в возврате.)
Я предполагаю, что это что-то довольно простое, что мне не хватает, но я не могу понять, что, какую-либо помощь очень ценили.
Ответы
Ответ 1
ok, я столкнулся с этой проблемой почти через 5 часов, пытаясь найти решение
то я нашел себя в исходном коде MVC.
и я обнаружил, что это проблема с исходным кодом Mvc в System.Web.Mvc.ValueProviderResult
на линии 173:
else if (valueAsArray != null)
{
// case 3: destination type is single element but source is array, so extract first element + convert
if (valueAsArray.Length > 0)
{
value = valueAsArray.GetValue(0);
return ConvertSimpleType(culture, value, destinationType);
}
else
{
// case 3(a): source is empty array, so can't perform conversion
return null;
}
}
поскольку вы можете видеть, является ли источник пустым массивом, он вернет null.
поэтому мне нужно найти способ обойти его, а затем я помню, как в старые добрые времена мы делали десериализацию:
вот как вы получите то, что хотите:
public ActionResult ProcessCriteria(int id, Criteria criteria)
{
var ser = new System.Web.Script.Serialization.JavaScriptSerializer();
StreamReader reader = new StreamReader(System.Web.HttpContext.Current.Request.InputStream);
reader.BaseStream.Position = 0;
criteria = ser.Deserialize<Criteria>(reader.ReadToEnd());
return Json(_service.ProcessCriteria(id, criteria));
}
Ответ 2
Вот что я написал в качестве комментария:
public class Criteria
{
public decimal? ANullableNumber { get; set; }
private IList<ObjectInList> _objectsList = new List<ObjectInList>();
public IList<ObjectInList> ObjectsList
{
get { return _objectsList; }
set {
if(value != null)
_objectsList = value;
}
}
}
Ответ 3
У меня есть ответ для вас, который будет работать на уровне структуры. В моем проекте я работал с данными, которые были немного больше, чем значения по умолчанию. Таким образом, я создал свой собственный ValueProviderFactory. Оказывается, если в массиве нет элементов, поставщик полностью пропускает эту запись. Вместо этого нам просто нужно сказать, что в массиве нет элементов. Вот код, который вам понадобится.
Во-первых, global.asax Application_Start:
public void Application_Start()
{
ValueProviderFactories.Factories.Remove(ValueProviderFactories.Factories.OfType<System.Web.Mvc.JsonValueProviderFactory>().FirstOrDefault());
ValueProviderFactories.Factories.Add(new LargeValueProviderFactory());
Во-вторых, вот вам другой класс:
#region << Usings >>
using System;
using System.Collections.Generic;
using System.Collections;
using System.Web.Mvc;
using System.IO;
using System.Web.Script.Serialization;
using System.Globalization;
#endregion
/// <summary>
/// This class is to ensure we can receive large JSON data from the client because the default is a bit too small.
/// </summary>
/// <remarks>This class is from the web.</remarks>
public sealed class LargeValueProviderFactory : System.Web.Mvc.ValueProviderFactory
{
#region << Constructors >>
/// <summary>
/// Default constructor.
/// </summary>
public LargeValueProviderFactory()
: base()
{
// Nothing to do
}
#endregion
#region << GetValueProvider >>
public override System.Web.Mvc.IValueProvider GetValueProvider(ControllerContext controllerContext)
{
if (controllerContext == null)
{
throw new ArgumentNullException("controllerContext");
}
object jsonData = GetDeserializedObject(controllerContext);
if (jsonData == null)
{
return null;
}
Dictionary<string, object> backingStore = new Dictionary<string, object>(StringComparer.OrdinalIgnoreCase);
AddToBackingStore(backingStore, String.Empty, jsonData);
return new DictionaryValueProvider<object>(backingStore, CultureInfo.CurrentCulture);
}
#endregion
#region << Helper Methods >>
private static void AddToBackingStore(Dictionary<string, object> backingStore, string prefix, object value)
{
IDictionary<string, object> d = value as IDictionary<string, object>;
if (d != null)
{
foreach (KeyValuePair<string, object> entry in d)
{
AddToBackingStore(backingStore, MakePropertyKey(prefix, entry.Key), entry.Value);
}
return;
}
IList l = value as IList;
if (l != null)
{
for (int i = 0; i < l.Count; i++)
{
AddToBackingStore(backingStore, MakeArrayKey(prefix, i), l[i]);
}
if (l.Count == 0)
backingStore[prefix] = value;
return;
}
// primitive
backingStore[prefix] = value;
}
private static object GetDeserializedObject(ControllerContext controllerContext)
{
if (!controllerContext.HttpContext.Request.ContentType.StartsWith("application/json", StringComparison.OrdinalIgnoreCase))
{
// not JSON request
return null;
}
StreamReader reader = new StreamReader(controllerContext.HttpContext.Request.InputStream);
string bodyText = reader.ReadToEnd();
if (String.IsNullOrEmpty(bodyText))
{
// no JSON data
return null;
}
JavaScriptSerializer serializer = new JavaScriptSerializer();
serializer.MaxJsonLength = Int32.MaxValue;
object jsonData = serializer.DeserializeObject(bodyText);
return jsonData;
}
private static string MakeArrayKey(string prefix, int index)
{
return prefix + "[" + index.ToString(CultureInfo.InvariantCulture) + "]";
}
private static string MakePropertyKey(string prefix, string propertyName)
{
return (String.IsNullOrEmpty(prefix)) ? propertyName : prefix + "." + propertyName;
}
#endregion
}
Ответ 4
Я думаю, что фактическая проблема заключается в DefaultModelBinder.cs строке 711, где он возвращает null, если встроенный objectList
не содержит ничего. Проверьте это: https://lostechies.com/jimmybogard/2013/11/07/null-collectionsarrays-from-mvc-model-binding/
Ответ 5
Это связано с тем, что вы никогда не определяете значение свойств NULL в классе "Критерии"; если он никогда не будет определен, он будет равен нулю.
например:
public class Criteria {
public decimal? ANullableNumber { get; set; }
public IList<ObjectInList> ObjectsList { get; set; }
}
public class Criteria1 {
private IList<ObjectInList> _ls;
private decimal? _num;
public decimal? ANullableNumber {
get {
if (_num == null) return 0;
return _num;
}
set {
_num = value;
}
}
public IList<ObjectInList> ObjectsList {
get {
if (_ls == null) _ls = new List<ObjectInList>();
return _ls;
}
set {
_ls = value;
}
}
}
public class HomeController : Controller {
public ActionResult Index() {
var dd = new Criteria();
return Json(dd); //output: {"ANullableNumber":null,"ObjectsList":null}
}
public ActionResult Index1() {
var dd = new Criteria1();
return Json(dd); //output: {"ANullableNumber":0,"ObjectsList":[]}
}
}