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

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 для сериализации ответа - нуль находится в объекте критериев, а не в возврате.)

Я предполагаю, что это что-то довольно простое, что мне не хватает, но я не могу понять, что, какую-либо помощь очень ценили.

4b9b3361

Ответ 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

}

Ответ 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":[]}
    }
  }