我正在使用VK API。有时服务器可以返回空数组而不是对象,例如:
personal: [] //when it is empty
或
personal: {
religion: 'Нет',
smoking: 1,
alcohol: 4
} //when not empty.
我正在使用JsonConvert反序列化大部分json。DeserializeObject,以及带有的json的这一部分
MainObject = ((MainObject["response"].GetObject())["user"].GetObject())["personal"].GetObject();
try
{
Convert.ToByte(MainObject["political"].GetNumber();
}
catch {}
但当应用程序处理大量的exeption时,它会使其工作缓慢。刚才我意识到,还有一些字段可能在为空时返回数组。我只是不知道如何快速而清晰地完成它。有什么建议吗?
我的反序列化类(字段为空时不起作用):
public class User
{
//some other fields...
public Personal personal { get; set; }
//some other fields...
}
public class Personal
{
public byte political { get; set; }
public string[] langs { get; set; }
public string religion { get; set; }
public string inspired_by { get; set; }
public byte people_main { get; set; }
public byte life_main { get; set; }
public byte smoking { get; set; }
public byte alcohol { get; set; }
}
另一个想法(不空时不起作用):
public List<Personal> personal { get; set; }
您可以制作一个JsonConverter
,如下所示,用于查找指定类型的对象或空数组。如果是对象,则反序列化该对象。如果是空数组,则返回null:
public class JsonSingleOrEmptyArrayConverter<T> : JsonConverter where T : class
{
public override bool CanConvert(Type objectType)
{
return typeof(T).IsAssignableFrom(objectType);
}
public override bool CanWrite { get { return false; } }
public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
{
var contract = serializer.ContractResolver.ResolveContract(objectType);
if (!(contract is Newtonsoft.Json.Serialization.JsonObjectContract || contract is Newtonsoft.Json.Serialization.JsonDictionaryContract))
{
throw new JsonSerializationException(string.Format("Unsupported objectType {0} at {1}.", objectType, reader.Path));
}
switch (reader.SkipComments().TokenType)
{
case JsonToken.StartArray:
{
int count = 0;
while (reader.Read())
{
switch (reader.TokenType)
{
case JsonToken.Comment:
break;
case JsonToken.EndArray:
return existingValue;
default:
{
count++;
if (count > 1)
throw new JsonSerializationException(string.Format("Too many objects at path {0}.", reader.Path));
existingValue = existingValue ?? contract.DefaultCreator();
serializer.Populate(reader, existingValue);
}
break;
}
}
// Should not come here.
throw new JsonSerializationException(string.Format("Unclosed array at path {0}.", reader.Path));
}
case JsonToken.Null:
return null;
case JsonToken.StartObject:
existingValue = existingValue ?? contract.DefaultCreator();
serializer.Populate(reader, existingValue);
return existingValue;
default:
throw new InvalidOperationException("Unexpected token type " + reader.TokenType.ToString());
}
}
public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
{
throw new NotImplementedException();
}
}
public static partial class JsonExtensions
{
public static JsonReader SkipComments(this JsonReader reader)
{
while (reader.TokenType == JsonToken.Comment && reader.Read())
;
return reader;
}
}
然后像这样使用:
public class User
{
//some other fields...
[JsonConverter(typeof(JsonSingleOrEmptyArrayConverter<Personal>))]
public Personal personal { get; set; }
//some other fields...
}
现在,您应该能够将用户反序列化到User
类中。
注:
转换器可以通过属性或在
JsonSerializerSettings.Converters
中应用。转换器不是为处理字符串等简单类型而设计的,它是为映射到JSON对象的类而设计的。这是因为它使用
JsonSerializer.Populate()
来避免读取过程中的无限递归。
工作样品。到处都是网络小提琴手。
不使用try-catch在两种可能性之间切换,只需检查第一个字符。如果是"[",则为null,如果是"{",则反序列化.
编辑:
现在,考虑到对象并不是JSON的全部,它给了我一个想法:我们在API返回不一致的JSON序列化时也遇到了类似的问题。最后,我们使用了NewtonSoft的ServiceStack。文本库(可从NuGet获得)。我们序列化为JToken对象,而不是目标类。然后,我们处理了JToken结构以进行逐段反序列化。