当值可以是对象或空数组时反序列化 JSON
Deserialize JSON when a value can be an object or an empty array
我正在使用 VK API。有时服务器可以 return 空数组而不是对象,例如:
personal: [] //when it is empty
或
personal: {
religion: 'Нет',
smoking: 1,
alcohol: 4
} //when not empty.
我用 JsonConvert.DeserializeObject 反序列化了 json 的大部分,json 的这一部分用
MainObject = ((MainObject["response"].GetObject())["user"].GetObject())["personal"].GetObject();
try
{
Convert.ToByte(MainObject["political"].GetNumber();
}
catch {}
但是当它处理大量异常时,它会使应用程序运行缓慢。刚才我意识到这里还有一些字段可能 return 数组为空时。我只是不知道如何快速而清晰地制作它。有什么建议吗?
我的反序列化 class(字段为空时不起作用):
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; }
不使用 try catch 在两种可能性之间切换,而是检查第一个字符。如果它是'[',它是null,如果它是'{'那么你反序列化。
编辑:
现在考虑到对象不是 JSON 的全部,它给了我一个想法:我们遇到了类似的问题 API 返回不一致的 JSON 序列化。最后,我们使用了 NewtonSoft 的 ServiceStack.Text 库(可从 NuGet 获得)。我们序列化为 JToken 对象而不是目标 class。然后我们处理 JToken 结构以进行分段反序列化。
您可以创建如下所示的 JsonConverter
,它查找指定类型的对象或空数组。如果是对象,则反序列化该对象。如果是空数组,它 returns 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
class.
备注:
可以通过属性或在JsonSerializerSettings.Converters
中应用转换器。
转换器并非设计用于处理字符串等简单类型,它是为映射到 JSON 对象的 classes 而设计的。那是因为它使用 JsonSerializer.Populate()
来避免读取过程中的无限递归。
我正在使用 VK API。有时服务器可以 return 空数组而不是对象,例如:
personal: [] //when it is empty
或
personal: {
religion: 'Нет',
smoking: 1,
alcohol: 4
} //when not empty.
我用 JsonConvert.DeserializeObject 反序列化了 json 的大部分,json 的这一部分用
MainObject = ((MainObject["response"].GetObject())["user"].GetObject())["personal"].GetObject();
try
{
Convert.ToByte(MainObject["political"].GetNumber();
}
catch {}
但是当它处理大量异常时,它会使应用程序运行缓慢。刚才我意识到这里还有一些字段可能 return 数组为空时。我只是不知道如何快速而清晰地制作它。有什么建议吗?
我的反序列化 class(字段为空时不起作用):
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; }
不使用 try catch 在两种可能性之间切换,而是检查第一个字符。如果它是'[',它是null,如果它是'{'那么你反序列化。
编辑:
现在考虑到对象不是 JSON 的全部,它给了我一个想法:我们遇到了类似的问题 API 返回不一致的 JSON 序列化。最后,我们使用了 NewtonSoft 的 ServiceStack.Text 库(可从 NuGet 获得)。我们序列化为 JToken 对象而不是目标 class。然后我们处理 JToken 结构以进行分段反序列化。
您可以创建如下所示的 JsonConverter
,它查找指定类型的对象或空数组。如果是对象,则反序列化该对象。如果是空数组,它 returns 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
class.
备注:
可以通过属性或在
JsonSerializerSettings.Converters
中应用转换器。转换器并非设计用于处理字符串等简单类型,它是为映射到 JSON 对象的 classes 而设计的。那是因为它使用
JsonSerializer.Populate()
来避免读取过程中的无限递归。