当通过JSON.NET序列化任意数据时,任何null属性都将作为
写入JSON“propertyName”:null
这当然是正确的。
但是我需要自动将所有空值转换为默认的空值,例如null string
应为String.Empty
,null int?
应为0
,null bool?
应为false
,依此类推。
NullValueHandling
没有帮助,因为我不想Ignore
空,但我也不想Include
他们(嗯,新功能?)。
所以我转向实施自定义JsonConverter
虽然实现本身是轻而易举的,但不幸的是,这仍然没有用 - CanConvert()
永远不会被调用具有空值的属性,因此也不会调用WriteJson()
。显然空值会自动序列化为null
,而不会自定义管道。
例如,以下是空字符串的自定义转换器示例:
public class StringConverter : JsonConverter
{
public override bool CanConvert(Type objectType)
{
return typeof(string).IsAssignableFrom(objectType);
}
...
public override void WriteJson(JsonWriter writer,
object value,
JsonSerializer serializer)
{
string strValue = value as string;
if (strValue == null)
{
writer.WriteValue(String.Empty);
}
else
{
writer.WriteValue(strValue);
}
}
}
在调试器中逐步执行此操作,我注意到这些方法都没有为具有空值的属性调用。
深入研究JSON.NET的源代码,我发现(显然,我没有深入研究)有一个特殊情况检查空值,并且明确地调用.WriteNull()
。
对于它的价值,我确实尝试实施自定义JsonTextWriter
并覆盖默认的.WriteNull()
实施...
public class NullJsonWriter : JsonTextWriter
{
...
public override void WriteNull()
{
this.WriteValue(String.Empty);
}
}
但是,这不能很好地工作,因为WriteNull()
方法对基础数据类型一无所知。当然,我可以为任何null输出""
,但这对于例如int,bool等。
所以,我的问题 - 没有手动转换整个数据结构,是否有任何解决方案或解决方法?
答案 0 :(得分:26)
好吧,我想我已经提出了一个解决方案(我的第一个解决方案根本不对,但我再次在火车上)。您需要为Nullable类型创建特殊合约解析程序和自定义ValueProvider。考虑一下:
public class NullableValueProvider : IValueProvider
{
private readonly object _defaultValue;
private readonly IValueProvider _underlyingValueProvider;
public NullableValueProvider(MemberInfo memberInfo, Type underlyingType)
{
_underlyingValueProvider = new DynamicValueProvider(memberInfo);
_defaultValue = Activator.CreateInstance(underlyingType);
}
public void SetValue(object target, object value)
{
_underlyingValueProvider.SetValue(target, value);
}
public object GetValue(object target)
{
return _underlyingValueProvider.GetValue(target) ?? _defaultValue;
}
}
public class SpecialContractResolver : DefaultContractResolver
{
protected override IValueProvider CreateMemberValueProvider(MemberInfo member)
{
if(member.MemberType == MemberTypes.Property)
{
var pi = (PropertyInfo) member;
if (pi.PropertyType.IsGenericType && pi.PropertyType.GetGenericTypeDefinition() == typeof (Nullable<>))
{
return new NullableValueProvider(member, pi.PropertyType.GetGenericArguments().First());
}
}
else if(member.MemberType == MemberTypes.Field)
{
var fi = (FieldInfo) member;
if(fi.FieldType.IsGenericType && fi.FieldType.GetGenericTypeDefinition() == typeof(Nullable<>))
return new NullableValueProvider(member, fi.FieldType.GetGenericArguments().First());
}
return base.CreateMemberValueProvider(member);
}
}
然后我用以下方法测试了它:
class Foo
{
public int? Int { get; set; }
public bool? Boolean { get; set; }
public int? IntField;
}
以下案例:
[TestFixture]
public class Tests
{
[Test]
public void Test()
{
var foo = new Foo();
var settings = new JsonSerializerSettings { ContractResolver = new SpecialContractResolver() };
Assert.AreEqual(
JsonConvert.SerializeObject(foo, Formatting.None, settings),
"{\"IntField\":0,\"Int\":0,\"Boolean\":false}");
}
}
希望这有点帮助...
编辑 - 更好地识别a Nullable<>
类型
编辑 - 添加了对字段和属性的支持,还可以在普通DynamicValueProvider
之上捎带以完成大部分工作,并使用更新的测试