经过大量的源代码挖掘,我解决了我的问题。原来所有在评论中提出的解决方案只适用于我反序列化一个包含字符串属性的复杂对象。在这种情况下,是的,简单修改合同解析器就可以解决[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].
}
当我们使用复杂对象时,JSON.NET会在内部将
TokenType
赋值为
JsonToken.StartObject
,这将导致反序列化按照特定路径进行,其中调用了
property.ValueProvider.SetValue(target, value);
。
然而,如果对象只是一个字符串,则
TokenType
将为
JsonToken.String
,路径也将不同,并且值提供程序将永远不会被调用。
无论如何,我的解决方案是构建一个自定义转换器来转换具有
TokenType == JsonToken.String
的
JsonReader
(下面是代码)。
解决方案:
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; }
}
}
[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);
}
}