I have a very undesirable situation which requires me to deserialize the JSON where the values are field names with JSON.NET. Assuming that I have the following JSON which i
You can create a custom JsonConverter
which serializes/deserializes Role[]
. You can then decorate your Roles
property with the JsonConverterAttribute
like this:
public class User
{
public string Name { get; set; }
[JsonConverter(typeof(RolesConverter))]
public Role[] Roles { get; set; }
}
In your converter class you are able to read an object and return an array instead. Your converter class may look like this:
class RolesConverter : JsonConverter
{
public override bool CanConvert(Type objectType)
{
return objectType == typeof(Role[]);
}
public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
{
// deserialize as object
var roles = serializer.Deserialize<JObject>(reader);
var result = new List<Role>();
// create an array out of the properties
foreach (JProperty property in roles.Properties())
{
var role = property.Value.ToObject<Role>();
role.Id = int.Parse(property.Name);
result.Add(role);
}
return result.ToArray();
}
public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
{
throw new NotImplementedException();
}
}
There are a few options available to you. You could have a custom JsonConverter
and serialize it manually. Since by the time of writing this fero provided an answer based on this, I'll give you an alternative, requiring two surrogate classes:
public class JsonUser
{
public string Name { get; set; }
public Dictionary<int, JsonRole> Roles { get; set; }
}
public class JsonRole
{
public string Name { get; set; }
}
And in your Role
class:
public static implicit operator User(JsonUser user)
{
return new User
{
Name = user.Name,
Roles = user.Roles
.Select(kvp => new Role { Id = kvp.Key, Name = kvp.Value.Name})
.ToArray()
};
}
Which can be used like this:
User jsonUser = JsonConvert.DeserializeObject<JsonUser>(json);
Now, this is done at the expense of creating an intermediate object and probably isn't suited for most cases.
For the sake of completeness, I'll include my version of the JsonConverter solution:
public class UserRolesConverter : JsonConverter
{
public override bool CanConvert(Type objectType)
{
return objectType == typeof (Role[]);
}
public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
{
return serializer.Deserialize<JObject>(reader)
.Properties()
.Select(p => new Role
{
Id = Int32.Parse(p.Name),
Name = (string) p.Value["name"]
})
.ToArray();
}
public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
{
throw new NotImplementedException();
}
}
public class User
{
public string Name { get; set; }
[JsonConverter(typeof(UserRolesConverter))]
public Role[] Roles { get; set; }
}
var jsonUser = JsonConvert.DeserializeObject<User>(json);
Hm you can try:
dynamic jObject = JObject.Parse(text);
List<User> users = new List<User>();
foreach(dynamic dUser in jObject)
{
List<Role> roles = new List<Role>();
User user = new User();
user.Name = dUser.name;
foreach(PropertyInfo info in dUser.GetType().GetProperties())
{
Role role = new Role();
role.Id = info.Name;
role.Name = dUser[info.Name].name;
roles.Ad(role);
}
user.Roles = roles.ToArray();
}