使用 Json.Net 将空字符串转换为 null

vic*_*tor 7 c# json.net

我找不到自动反序列化(服务器端)所有EmptyOrWhiteSpace字符串到null. Json.Net 默认只是简单地将值赋值给对象属性,我需要逐个字符串验证它是空的还是空白的,然后将其设置为 null。

我需要在反序列化时完成此操作,因此我不必记住验证来自客户端的每个字符串。

如何在 Json Net 上覆盖它?

vic*_*tor 6

经过大量的源代码挖掘,我解决了我的问题。事实证明,评论中提出的所有解决方案仅在我反序列化包含字符串属性的复杂对象时才有效。在这种情况下,是的,只需修改合同解析器即可工作 [1]。

但是,我需要的是一种在反序列化时将任何字符串转换为 null 的方法,并且在我的对象只是一个字符串的情况下,以这种方式修改合同将失败,即,

public void MyMethod(string jsonSomeInfo)
{
  // At this point, jsonSomeInfo is "\"\"",
  // an emmpty string.

  var deserialized = new JsonSerializer().Deserialize(new StringReader(jsonSomeInfo), typeof(string));

  // deserialized = "", event if I used the modified contract resolver [1].
}
Run Code Online (Sandbox Code Playgroud)

发生的情况是,当我们处理一个复杂对象时,JSON.NET 内部会为读取器分配一个TokenTypeof JsonToken.StartObject,这将导致反序列化遵循property.ValueProvider.SetValue(target, value);调用的特定路径。

但是,如果对象只是一个字符串,则TokenTypewillJsonToken.String和 path 将不同,并且永远不会调用值提供程序。

无论如何,我的解决方案是构建一个自定义转换器来转换JsonReader具有TokenType == JsonToken.String(下面的代码)的s 。

解决方案

public class StringConverter : JsonConverter
{
    public override bool CanConvert(Type objectType)
    {
        return objectType == typeof(string);
    }

    public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
    {
            if (reader.Value == null) return null;

            string text = reader.Value.ToString();

            if (string.IsNullOrWhiteSpace(text))
            {
                return null;
            }

            return text;
    }

    public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
    {
        throw new NotImplementedException("Not needed because this converter cannot write json");
    }

    public override bool CanWrite
    {
        get { return false; }
    }
}
Run Code Online (Sandbox Code Playgroud)

[1] 感谢@Raphaël Althaus。

public class NullToEmptyStringResolver : DefaultContractResolver
{
    protected override IList<JsonProperty> CreateProperties(Type type, MemberSerialization memberSerialization)
    {
        return type.GetProperties()
        .Select(p => {
            var jp = base.CreateProperty(p, memberSerialization);
            jp.ValueProvider = new EmptyToNullStringValueProvider(p);
            return jp;
        }).ToList();
    }
}

public class EmptyToNullStringValueProvider : IValueProvider
{
    PropertyInfo _MemberInfo;

    public EmptyToNullStringValueProvider(PropertyInfo memberInfo)
    {
        _MemberInfo = memberInfo;
    }

    public object GetValue(object target)
    {
        object result = _MemberInfo.GetValue(target);

        if (_MemberInfo.PropertyType == typeof(string) && result != null && string.IsNullOrWhiteSpace(result.ToString()))
        {
            result = null;
        }

        return result;
    }

    public void SetValue(object target, object value)
    {
        if (_MemberInfo.PropertyType == typeof(string) && value != null && string.IsNullOrWhiteSpace(value.ToString()))
        {
            value = null;
        }

        _MemberInfo.SetValue(target, value);
    }
}
Run Code Online (Sandbox Code Playgroud)