nik*_*_10 3 c# json json.net vk
我正在使用 VK API。有时服务器可以返回空数组而不是对象,例如:
\n\npersonal: [] //when it is empty\nRun Code Online (Sandbox Code Playgroud)\n\n或者
\n\npersonal: {\nreligion: \'\xd0\x9d\xd0\xb5\xd1\x82\',\nsmoking: 1,\nalcohol: 4\n} //when not empty.\nRun Code Online (Sandbox Code Playgroud)\n\n我正在使用 JsonConvert.DeserializeObject 反序列化大部分 json,而这部分 json 则使用
\n\nMainObject = ((MainObject["response"].GetObject())["user"].GetObject())["personal"].GetObject();\ntry\n{\nConvert.ToByte(MainObject["political"].GetNumber();\n} \ncatch {}\nRun Code Online (Sandbox Code Playgroud)\n\n但当应用程序处理大量异常时,它会使应用程序运行缓慢。刚才我意识到还有一些字段在为空时可能会返回数组。我只是不知道如何快速而清晰地做到这一点。有什么建议么?
\n\n我的反序列化类(当字段为空时不起作用):
\n\n public class User\n {\n//some other fields...\n public Personal personal { get; set; }\n//some other fields...\n }\n public class Personal\n {\n public byte political { get; set; }\n public string[] langs { get; set; }\n public string religion { get; set; }\n public string inspired_by { get; set; }\n public byte people_main { get; set; }\n public byte life_main { get; set; }\n public byte smoking { get; set; }\n public byte alcohol { get; set; }\n }\nRun Code Online (Sandbox Code Playgroud)\n\n另一个想法(非空时不起作用):
\n\npublic List<Personal> personal { get; set; }\nRun Code Online (Sandbox Code Playgroud)\n
您可以创建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;
}
}
Run Code Online (Sandbox Code Playgroud)
然后像这样使用它:
public class User
{
//some other fields...
[JsonConverter(typeof(JsonSingleOrEmptyArrayConverter<Personal>))]
public Personal personal { get; set; }
//some other fields...
}
Run Code Online (Sandbox Code Playgroud)
您现在应该能够将用户反序列化到您的User类中。
笔记:
转换器可以通过属性或在JsonSerializerSettings.Converters.
该转换器不适用于简单类型(例如字符串),它是为映射到 JSON 对象的类而设计的。这是因为它用于JsonSerializer.Populate()避免读取过程中的无限递归。