Преобразование 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 из API REST.
То, что я на самом деле пытаюсь выполнить, - написать общий метод, который может легко отобразить мой ответ JSON в мои объекты CLR, как вы уже можете сделать с RestSharp. Единственная проблема заключается в том, что реализация по умолчанию RestSharp не работает для меня, и она не может успешно разобрать JSON, поскольку ответ не всегда возвращает все свойства (я не возвращаю поля, которые null
с сервера REST).
Вот почему я решил использовать Newtonsoft Json.NET, так как он имеет гораздо более мощный Json-десериализационный движок. К сожалению, он не поддерживает имена нечетких свойств/полей, такие как RestSharp (или я их не нашел), поэтому он также неправильно отображает мои объекты CLR, когда я использую что-то вроде say JsonConvert.DeserializeObject<User>(response.Content)
.
Вот как выглядит мой Json (пример на самом деле):
{
"id" : 77239923,
"username" : "UzEE",
"email" : "[email protected]",
"name" : "Uzair Sajid",
"twitter_screen_name" : "UzEE",
"join_date" : "2012-08-13T05:30:23Z05+00",
"timezone" : 5.5,
"access_token" : {
"token" : "nkjanIUI8983nkSj)*#)([email protected]",
"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 Response объекту-объекту пользователя. Вот фактический фрагмент кода того, что я делаю в методе расширения 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(и, честно говоря, документация любопытная, хотя для понимания и отсутствия некоторых примеров).
Любая помощь будет действительно оценена.
Ответы
Ответ 1
Теперь существует метод ToObject.
var obj = jsonObject["date_joined"];
var result = obj.ToObject<DateTime>();
Он также работает с любым сложным типом и подчиняется правилам JsonPropertyAttribute
var result = obj.ToObject<MyClass>();
public class MyClass
{
[JsonProperty("date_field")]
public DateTime MyDate {get;set;}
}
Ответ 2
System.Convert.ChangeType(jtoken.ToString(), targetType);
или
JsonConvert.DeserializeObject(jtoken.ToString(), targetType);
- EDIT -
Узайр. Вот полный пример, чтобы показать вам, что они работают.
string json = @"{
""id"" : 77239923,
""username"" : ""UzEE"",
""email"" : ""[email protected]"",
""name"" : ""Uzair Sajid"",
""twitter_screen_name"" : ""UzEE"",
""join_date"" : ""2012-08-13T05:30:23Z05+00"",
""timezone"" : 5.5,
""access_token"" : {
""token"" : ""nkjanIUI8983nkSj)*#)([email protected]"",
""scope"" : [ ""read"", ""write"", ""bake pies"" ],
""expires"" : 57723
},
""friends"" : [{
""id"" : 2347484,
""name"" : ""Bruce Wayne""
},
{
""id"" : 996236,
""name"" : ""Clark Kent""
}]
}";
var obj = (JObject)JsonConvert.DeserializeObject(json);
Type type = typeof(int);
var i1 = System.Convert.ChangeType(obj["id"].ToString(), type);
var i2 = JsonConvert.DeserializeObject(obj["id"].ToString(), type);
Ответ 3
var i2 = JsonConvert.DeserializeObject(obj["id"].ToString(), type);
вызывает исключение синтаксического анализа из-за отсутствия котировок вокруг первого аргумента (я думаю). Я получил его для работы, добавив цитаты:
var i2 = JsonConvert.DeserializeObject("\"" + obj["id"].ToString() + "\"", type);
Ответ 4
Мне удалось преобразовать, используя метод ниже для моего WebAPI:
[HttpPost]
public HttpResponseMessage Post(dynamic item) // Passing parameter as dynamic
{
JArray itemArray = item["Region"]; // You need to add JSON.NET library
JObject obj = itemArray[0] as JObject; // Converting from JArray to JObject
Region objRegion = obj.ToObject<Region>(); // Converting to Region object
}