JSon.Net特定JSonConverter问题,必须忽略空数组

问题描述:

我注意到序列化中的怪异行为。 虽然,我有设置JSon.Net特定JSonConverter问题,必须忽略空数组

var SerializerSettings = new JsonSerializerSettings() { 
        NullValueHandling = NullValueHandling.Ignore,  DefaultValueHandling= DefaultValueHandling.Ignore} 
SerializerSettings.Converters.Add(new JsonArrayToNullConverter()); 

    var val = new company() 
    { 
     name = "Bobo Company Renamed" 
    } 
var str = JsonConvert.SerializeObject(val, SerializerSettings); 

其结果将是: { “DOCUMENT_TYPE”:2, “位置”:空...

没有自定义转换器,这将是 {“document_type”:2,“locations”:[],...

你明白了吗?

但是,因为它变成零,它-should-听NullValueHandling = NullValueHandling.Ignore 但显然,牛顿,着眼于对象的值被序列化,而不是在发出writer.WriteNull();

:(

任何解决方法吗?我已经花了这一点。感谢几个小时!

using Newtonsoft.Json; 

using System; 
using System.Collections; 
using System.Collections.Generic; 

namespace Company.Model.TypeConverters 
{ 
    /// <summary> 
    /// undo's the forced array creation because OData needs it, but for PATCH, we want no [] empty collections, it is considered to be a 
    /// </summary> 
    public class JsonArrayToNullConverter : JsonConverter 
    { 

     public override bool CanConvert(Type objectType) 
     { 
      var canConvert = objectType.IsArray || (objectType.IsGenericType && objectType.GetGenericTypeDefinition().IsAssignableFrom(typeof(IEnumerable<>))); 
      return canConvert; 
     } 
     public override bool CanRead 
     { 
      get 
      { 
       return false; 
      } 
     } 
     public override bool CanWrite 
     { 
      get 
      { 
       return true; 
      } 
     } 
     public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer) 
     { 
      throw new NotImplementedException(); 
     } 

     public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer) 
     { 
      var enumerator = (IEnumerable)value; 

      if (enumerator != null && enumerator.GetEnumerator().MoveNext() == false) 
      { 
       writer.WriteNull(); 
       //value = null; 
       return; 
      } 
      serializer.Serialize(writer, value); 
     } 
    } 
} 

的解决方案似乎是使用contractresolver,并设置在JsonSerializerSettings。

public class NullToEmptyListResolver : DefaultContractResolver 
{ 
    protected override IValueProvider CreateMemberValueProvider(MemberInfo member) 
    { 
     IValueProvider provider = base.CreateMemberValueProvider(member); 

     if (member.MemberType == MemberTypes.Property) 
     { 
      Type propType = ((PropertyInfo)member).PropertyType; 
      if (propType.IsArray || (propType.IsGenericType && 
       propType.GetGenericTypeDefinition().IsAssignableFrom(typeof(IEnumerable<>)))) 
      { 
       return new EmptyListValueProvider(provider, propType); 
      } 
     } 

     return provider; 
    } 

    public class EmptyListValueProvider : IValueProvider 
    { 
     private readonly IValueProvider innerProvider; 

     public EmptyListValueProvider(IValueProvider innerProvider, Type listType) 
     { 
      this.innerProvider = innerProvider; 
     } 

     public void SetValue(object target, object value) 
     { 
      throw new NotImplementedException(); 
     } 

     public object GetValue(object target) 
     { 
      var val = innerProvider.GetValue(target) ; 
      if (val == null) return null; 
      var enumerator = (IEnumerable)val; 
      return enumerator.GetEnumerator().MoveNext() == false ? null : val; 
     } 
    } 
+0

在你的'SetValue()'方法中,你应该调用'innerProvider.SetValue(target,value);'而不是抛出一个异常,并且考虑[处置枚举器](https://stackoverflow.com/q/232558/37441 82)使用后。例如。 [这个答案](https://stackoverflow.com/a/24706061/3744182)隐含。 – dbc

+0

谢谢,但为了我的目的,它不能写,只能读。 –