当值可以是对象或空数组时,反序列化JSON



我正在使用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结构以进行逐段反序列化。

相关内容

  • 没有找到相关文章

最新更新