我在.NET 4.7中使用JSON.Net。
在我调用的API中,我可以为同一个调用看似随机地获得两种错误json:(。
错误类型1:
{
"Code": 0,
"Msg": ["Some Warning"]
}
错误类型2:
{
"Code": 0,
"Msg": [
{"Error": 5 },
{"Error": 6 }
]
}
如您所见,Msg
更改了其结构。我想使用通用的POCO反序列化,但是Msg
会如何变化?
JsonConvert.DeserializeObject<MyCommonPoco>(theJson);
答案 0 :(得分:4)
您可以write a custom JsonConverter
处理不同的类型。例如,您可以使用以下JSON:
{
"Code": 0,
"Msg": [
"A single string",
{ "Message": "An object with a message" },
{ "Message": "An object with a message and a code", "Code": 5 },
{ "Code": 5 }
]
}
在此示例中,Msg
数组可以包含原始字符串或复杂对象。并且该对象可以包含Message
,Code
或两者。
您将为此选择最通用的数据结构,该数据结构能够表示所有这些信息。例如,这意味着您必须将原始字符串包装到兼容的对象中。
public class Error
{
public int Code { get; set; }
public ErrorMessage[] Msg { get; set; }
}
public class ErrorMessage
{
public string Message { get; set; }
public int? Code { get; set; }
}
然后您可以使用自定义
然后,使用自定义JsonConverter
,您可以解析上述JSON:
public class ErrorMessageConverter : JsonConverter
{
public override bool CanConvert(Type objectType)
=> objectType == typeof(ErrorMessage);
public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
{
// handle the primitive string
if (reader.TokenType == JsonToken.String)
return new ErrorMessage { Message = (string)reader.Value };
// handle a complex object; just assume that it is an `ErrorMessage` here
else if (reader.TokenType == JsonToken.StartObject)
return JObject.Load(reader).ToObject<ErrorMessage>();
throw new NotSupportedException();
}
public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
{
throw new NotImplementedException();
}
}
这将是使用该转换器的一种方式:
// the single quotes are just here to make it more readable; JSON.NET support this
// although it isn’t valid JSON
var json = @"{
'Code': 0,
'Msg': [
'A single string',
{ 'Message': 'An object with a message' },
{ 'Message': 'An object with a message and a code', 'Code': 5 },
{ 'Code': 5 }
]
}";
var settings = new JsonSerializerSettings()
settings.Converters.Add(new ErrorMessageConverter());
var result = JsonConvert.DeserializeObject<Error>(json, settings);