在我的WPF代码中,我使用Newtonsoft。Json来反序列化Json到我的模型。首先,我收到一个Json字符串(' Json '),然后我将其解析为'message'。(我想要反序列化的对象被包装在json字符串中的"data"字段中)。
Activity message = JObject.Parse(json)["data"].ToObject<Activity>();
我的Activity类使用几个[JsonProperty]属性来生成它的字段。其中之一是名为"ActivityType"的enum。
[JsonProperty("type")]
[JsonConverter(typeof(ActivityTypeConverter))]
public ActivityType Type { get; set; }
public enum ActivityType {
EmailOpen,
LinkClick,
Salesforce,
Unsupported
};
public class ActivityTypeConverter : JsonConverter {
public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
{
var rawString = existingValue.ToString().ToLower();
if (rawString.Contains("click"))
return ActivityType.LinkClick;
else if (rawString.Contains("salesforce"))
return ActivityType.Salesforce;
else if (rawString.Contains("email_open"))
return ActivityType.EmailOpen;
else
{
Console.WriteLine("unsupported " + rawString);
return ActivityType.Unsupported;
}
}
public override bool CanConvert(Type objectType)
{
return !objectType.Equals(typeof(ActivityType));
}
public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
{
throw new NotImplementedException();
}
}
什么是奇怪和令人沮丧的是json对象,我知道有"类型":"email_open"被反序列化为ActivityType。不支持,即使我的转换器应该将它们反序列化为EmailOpen。
调试显示了问题所在:json字段"type"自动反序列化"email_open"为EmailOpen ,然后通过我的转换器发送。(它中断了,因为我的条件检查下划线,而EmailOpen.ToString()没有。)
所以我的问题是:为什么它在没有我的转换器的情况下转换,我如何阻止它?我只想让它只使用我的转换器
我认为您的转换器正在被调用-它只是不工作。问题是,不是从JsonReader reader
读取新值,而是使用来自existingValue
的值。但是第二个值是被反序列化的类中已经存在的属性值,而不是被读取的值。
您需要沿着Json行从读取器加载值。. NET的StringEnumConverter
。这里有一个版本可以做到这一点,并且还可以通过子类化StringEnumConverter
并将从文件读取的值传递到基类以进行进一步处理来处理枚举的标准值:
public class ActivityTypeConverter : StringEnumConverter
{
public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
{
bool isNullable = (Nullable.GetUnderlyingType(objectType) != null);
Type type = (Nullable.GetUnderlyingType(objectType) ?? objectType);
if (reader.TokenType == JsonToken.Null)
{
if (!isNullable)
throw new JsonSerializationException();
return null;
}
var token = JToken.Load(reader);
if (token.Type == JTokenType.String)
{
var rawString = ((string)token).ToLower();
if (rawString.Contains("click"))
return ActivityType.LinkClick;
else if (rawString.Contains("salesforce"))
return ActivityType.Salesforce;
else if (rawString.Contains("email_open"))
return ActivityType.EmailOpen;
}
using (var subReader = token.CreateReader())
{
while (subReader.TokenType == JsonToken.None)
subReader.Read();
try
{
return base.ReadJson(subReader, objectType, existingValue, serializer); // Use base class to convert
}
catch (Exception ex)
{
return ActivityType.Unsupported;
}
}
}
public override bool CanConvert(Type objectType)
{
return objectType == typeof(ActivityType);
}
}