Преобразование JToken (или строки )в заданный тип

Версия TL;DR

У меня есть объект типаJToken(но также может бытьstring)и мне нужно преобразовать его в тип, содержащийся в переменной type:

Type type = typeof(DateTime); /* can be any other Type like string, ulong etc */
var obj = jsonObject["date_joined"]; /* contains 2012-08-13T06:01:23Z+05:00 */
var result = Some_Way_To_Convert(type, obj);

Приведенный выше resultдолжен быть объектом DateTime со значением, указанным в date_joined.

Полная история

Я использую как RestSharp, так и Json.NET в проекте Windows Phone, и я застрял, пытаясь десериализовать ответы JSON из REST API.

Что я на самом деле пытаюсь сделать, так это написать общий метод, который может легко отображать мой ответ JSON в мои объекты CLR, так же, как вы уже можете сделать с RestSharp. Единственная проблема заключается в том, что реализация RestSharp по умолчанию не работает для меня и не может успешно проанализировать JSON, поскольку ответ не всегда возвращает все свойства (. Я не возвращаю поля, которые nullиз REST-сервер ).

Вот почему я решил использовать Json.NET от Newtonsoft, так как он имеет гораздо более мощный механизм десериализации Json. К сожалению, он не поддерживает нечеткие имена свойств/полей, такие как RestSharp (, или я не нашел таких ),поэтому он также неправильно сопоставляется с моими объектами CLR, когда я использую что-то вроде say JsonConvert.DeserializeObject<User>(response.Content).

Вот как выглядит мой Json (пример на самом деле):

{
    "id" : 77239923,
    "username" : "UzEE",
    "email" : "uzee@email.net",
    "name" : "Uzair Sajid",
    "twitter_screen_name" : "UzEE",
    "join_date" : "2012-08-13T05:30:23Z05+00",
    "timezone" : 5.5,
    "access_token" : {
        "token" : "nkjanIUI8983nkSj)*#)(kjb@K",
        "scope" : [ "read", "write", "bake pies" ],
        "expires" : 57723
    },
    "friends" : [{
        "id" : 2347484",
        "name" : "Bruce Wayne"
    },
    {
        "id" : 996236,
        "name" : "Clark Kent"
    }]
}

А вот пример моих сущностей CLR:

class AccessToken 
{
    public string Token { get; set; }
    public int Expires { get; set; }
    public string[] Scope { get; set; }
    public string Secret { get; set; } /* may not always be returned */
}

class User
{
    public ulong Id { get; set; }
    public string UserName { get; set; }
    public string Email { get; set; }
    public string Name { get; set; }
    public string TwitterScreenName { get; set; }
    public DateTime JoinDate { get; set; }
    public float Timezone { get; set; }
    public bool IsOnline { get; set; } /* another field that might be blank e.g. */

    public AccessToken AccessToken { get; set; }

    public List<User> Friends { get; set; }
}

Мне нужен простой способ разобрать приведенный выше JSON на заданные объекты CLR. Я просмотрел исходный код RestSharp и увидел код JsonDeserializer, и мне удалось написать универсальный метод расширения DeserializeResponse<T>для JObject, который должен возвращать объект типа T. Предполагаемое использование примерно такое:

var user = JObject.Parse(response.Content).DeserializeResponse<User>();

Вышеупомянутый метод должен анализировать данный ответ Json на объект сущности пользователя. Вот фактический фрагмент кода того, что я делаю в методе расширения DeserializeResponse<User>(, основанном на коде RestSharp ):

public static T DeserializeResponse<T>(this JObject obj) where T : new()
{
    T result = new T();
    var props = typeof(T).GetProperties().Where(p => p.CanWrite).ToList();
    var objectDictionary = obj as IDictionary<string, JToken>;

    foreach (var prop in props)
    {
        var name = prop.Name.GetNameVariants(CultureInfo.CurrentCulture).FirstOrDefault(n => objectDictionary.ContainsKey(n));
        var value = name != null ? obj[name] : null;

        if (value == null) continue;

        var type = prop.PropertyType;

        if (type.IsGenericType && type.GetGenericTypeDefinition() == typeof(Nullable<>))
        {
            type = type.GetGenericArguments()[0];
        }

        // This is a problem. I need a way to convert JToken value into an object of Type type
        prop.SetValue(result, ConvertValue(type, value), null); 
    }

    return result;
}

. Я предполагаю, что преобразование должно быть очень простым делом, поскольку это тривиальная задача. Но я искал довольно долго и до сих пор не нашел способа сделать это через Json.NET (и, будем честными, документация довольно понятная и не хватает некоторых примеров ).

Любая помощь будет действительно оценена.

64
задан Uzair Sajid 13 August 2012 в 01:05
поделиться