25

我有一个控制器动作,它接收一个整数和一个对象,其中包含各种属性,其中一个是对象的通用列表。当我使用填充列表将 JSON 发布到操作时,所有内容都正确映射,并且我得到一个包含我发布的对象的列表。但是,如果数组为空,则 MVC 操作将属性绑定到空列表而不是空列表。我希望空数组映射到一个空数组而不是 null,因为在这种情况下,空数组意味着集合中没有任何内容,而 null 意味着应该检查数据库以查看以前是否有任何内容保存在集合中,但我不知道需要更改什么才能使其正确映射。我们正在使用 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 请求:"{\"id\":137,\"criteria\":{\"ObjectsList\":[]}}"

控制器动作:

public ActionResult ProcessCriteria(int id, Criteria criteria)
{
    return Json(_service.ProcessCriteria(id, criteria));
}

正是在控制器操作中,我在标准对象中得到了一个空列表而不是一个空列表。无论我是否为其他属性发送空值,都会发生这种情况。不确定是否归结为对象是 IList 而不是 IEnumerable?(包装服务调用的 Json 方法是我们使用 Json.Net 返回 json 结果以序列化响应的包装器 - null 在接收到的条件对象中,而不是在返回中。)

我猜这是我错过的非常简单的事情,但我无法弄清楚是什么,非常感谢任何帮助。

4

6 回答 6

12

好的,我面临这个问题将近 5 个小时试图找到解决方案,然后我发现自己正在查看 MVC 源代码。我发现这是 第 173 行System.Web.Mvc.ValueProviderResult中的 Mvc 源代码的问题:

        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;
            }
        }

如您所见,如果 source 为空数组,它将返回 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));
    }
于 2013-03-28T01:40:53.070 回答
2

解决此问题的一种方法是将新实例分配为您的默认值,ObjectsList如下所示:

public class Criteria
{
    public decimal? ANullableNumber { get; set; }
    public IList<ObjectInList> ObjectsList { get; set; } = new List<ObjectInList>();
}

List如果 JSON 数组中null没有值,这将创建一个空值。

于 2019-02-18T16:46:01.710 回答
1

我有一个适用于框架级别的答案。在我的项目中,我使用的数据比默认值支持的数据要大一些。因此,我创建了自己的 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());

其次,这是您需要的其他课程:

using System;
using System.Collections.Generic;
using System.Collections;
using System.Web.Mvc;
using System.IO;
using System.Web.Script.Serialization;
using System.Globalization;

public sealed class LargeValueProviderFactory : System.Web.Mvc.ValueProviderFactory
{    
    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);
    }

    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;
    }
}
于 2013-08-04T14:54:08.197 回答
1

我认为实际问题出在DefaultModelBinder.cs第 711 行,如果构建的objectList内容不包含任何内容,它将返回 null。看看这个:https ://lostechies.com/jimmybogard/2013/11/07/null-collectionsarrays-from-mvc-model-binding/

于 2017-01-30T02:40:36.147 回答
0

这是我作为评论发布的内容:

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;
        }
     }
}
于 2013-01-23T12:51:55.900 回答
-2

这是因为您从未在“Criteria”类中定义可为空的属性值;如果从未定义,它将为空。

例如:

  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":[]}
    }
  }
于 2013-11-08T09:46:28.493 回答