19

我有一种情况,JSONREST服务返回的返回一个电影对象列表,所有这些都包含大量信息。该服务结果中的几个字段REST会根据可用信息而变化。

一个例子:一部电影总是有一些屏幕截图(图像)、演员和导演。根据所讨论的电影,可能有一个或多个图像、一个或多个演员和一个或多个导演。几个案例的示例 JSON:

{
    "title": "Movie title",
    "images": [
        "http://www.url.com/img_0.jpg",
        "http://www.url.com/img_1.jpg",
        "http://www.url.com/img_2.jpg",
        "http://www.url.com/img_3.jpg",
        "http://www.url.com/img_4.jpg"
    ],
    "actors": [
        "Steven Berkoff",
        "Nikolaj Coster-Waldau",
        "Julie Cox"
    ],
    "directors": "Simon Aeby"
},
{
    "title": "Another movie",
    "images": "http://www.url.com/img_1.jpg",
    "actors": "actor 1"
    "directors": [
        "Justin Bieber",
        "Justin Timberlake"
    ]
}

问题是,使用 JSON.net,我怎样才能创建一个转换器来处理这个问题?我一直在网上搜索,但仍然没有找到解决方案。

同一个问题的另一个旋转:如果一个字段是字符串列表或简单字符串,我如何让 JSON.NET 以任何一种方式创建一个列表(如果只是一个简单的字符串,则创建一个包含一个成员的列表)

编辑:这个 REST 服务是我无法控制的

4

2 回答 2

11

好吧,我这样做是为了好玩,但无论如何,不​​要认为有用或最好的方法......

将“动态”属性声明为对象,然后创建获取属性的方法,例如 ImagesAsList 或 ImagesAsString。我用扩展方法做到了......

var movies = JsonConvert.DeserializeObject<List<Movie>>(str);

班级

class Movie
{

    [JsonProperty("title")]
    public string Title { get; set; }

    [JsonProperty("images")]
    public object Images { get; set; }

    [JsonProperty("actors")]
    public object Actor { get; set; }

    [JsonProperty("directors")]
    public object Directors { get; set; }
}

扩展方法

static class MovieExtension
{
    public static List<string> ImagesAsList(this Movie m)
    {
        var jArray = (m.Images as JArray);
        if (jArray == null) return null;

        return jArray.Select(x => x.ToString()).ToList();
    }

    public static string ImagesAsString(this Movie m)
    {
        return m.Images as string;
    }

}

编辑

阅读@yamen 评论后,我做了一些更改,例如:

var settings = new JsonSerializerSettings();
settings.Converters.Add(new MoviesConverter());

var movies = JsonConvert.DeserializeObject<List<Movie>>(str, settings);

班级

class Movie
{

    [JsonProperty("title")]
    public List<string> Title { get; set; }

    [JsonProperty("images")]
    public List<string> Images { get; set; }

    [JsonProperty("actors")]
    public List<string> Actor { get; set; }

    [JsonProperty("directors")]
    public List<string> Directors { get; set; }
}

转换器

class MoviesConverter : JsonConverter
{
    public override bool CanConvert(Type objectType)
    {
        return (objectType == typeof(string)) || (objectType == typeof(List<string>)) ;
    }

    public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
    {
        if (reader.TokenType == JsonToken.StartArray)
        {
            var l = new List<string>();
            reader.Read();
            while (reader.TokenType != JsonToken.EndArray)
            {
                l.Add(reader.Value as string);

                reader.Read();
            }
            return l;
        }
        else
        {
            return new List<string> { reader.Value as string };
        }
    }

    public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
    {
        //ToDo here we can decide to write the json as 
        //if only has one attribute output as string if it has more output as list
    }
}
于 2012-04-12T11:06:45.907 回答
4

您将无法直接序列化到对象,但您可以手动执行此操作而无需太多努力。JSON.Net 包含 LINQ to JSON。首先定义一个方法,即使底层的 JSON 不是一个数组,它也总是返回一个类型为 T 的列表:

public List<T> getSingleOrArray<T>(JToken token)
{
    if (token.HasValues)
    {
        return token.Select(m => m.ToObject<T>()).ToList();
    }
    else
    {
        return new List<T> { token.ToObject<T>() };
    }
}

示例用法:

JObject m1 = JObject.Parse(@"{
""title"": ""Movie title"",
""images"": [
    ""http://www.url.com/img_0.jpg"",
    ""http://www.url.com/img_1.jpg""
],
""actors"": [
    ""Steven Berkoff"",
    ""Julie Cox""
],
""directors"": ""Simon Aeby""
}");

JObject m2 = JObject.Parse(@"{
""title"": ""Another movie"",
""images"": ""http://www.url.com/img_1.jpg"",
""actors"": ""actor 1"",
""directors"": [
    ""Justin Bieber"",
    ""Justin Timberlake""
]
}");

IList<String> m1_directors = getSingleOrArray<string>(m1["directors"]);
IList<String> m2_directors = getSingleOrArray<string>(m2["directors"]);

m1_directory 是一个包含单个元素的列表,m2_directors 是一个包含两个元素的列表。

于 2012-04-12T11:05:03.237 回答