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

nik*_*_10 3 c# json json.net vk

我正在使用 VK API。有时服务器可以返回空数组而不是对象,例如:

\n\n
personal: [] //when it is empty\n
Run Code Online (Sandbox Code Playgroud)\n\n

或者

\n\n
personal: {\nreligion: \'\xd0\x9d\xd0\xb5\xd1\x82\',\nsmoking: 1,\nalcohol: 4\n} //when not empty.\n
Run Code Online (Sandbox Code Playgroud)\n\n

我正在使用 JsonConvert.DeserializeObject 反序列化大部分 json,而这部分 json 则使用

\n\n
MainObject = ((MainObject["response"].GetObject())["user"].GetObject())["personal"].GetObject();\ntry\n{\nConvert.ToByte(MainObject["political"].GetNumber();\n} \ncatch {}\n
Run 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            }\n
Run Code Online (Sandbox Code Playgroud)\n\n

另一个想法(非空时不起作用):

\n\n
public List<Personal> personal { get; set; }\n
Run Code Online (Sandbox Code Playgroud)\n

dbc*_*dbc 5

您可以创建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类中。

笔记:

工作示例 .Net在此处此处进行操作。