json.net: указать конвертер для ключей словаря
У меня есть JSON:
{ "data": { "A": 5, "B": 6 }, "foo": "foo", "bar": "bar" }
Мне нужно десериализовать данные в class:
public Dictionary Data { get; set; } public string Foo { get; set; } public string Bar { get; set; }
Но значения MyEnum – это CodeA
и CodeB
а не просто A
и B
соответственно.
- Строка Parse Json в C #
- Как десериализовать JSON с двойными именами свойств в одном и том же объекте
- Невозможно десериализовать текущий объект JSON (например, {"name": "value"}) в тип 'System.Collections.Generic.List`1
- Deserializing JSON, когда иногда массивы, а иногда и объекты
- Json.NET Отключить десериализацию в DateTime
У меня есть собственный конвертер, который может обрабатывать преобразование. Но как я могу указать JsonConverter
для использования с ключами словаря?
- Как добавить дополнительное свойство в сериализованную строку JSON с помощью json.net?
- JSON.NET как сериализатор OAP для WebAPI 2 и ODataMediaTypeFormatter
- C # WCF REST. Как вы используете сериализатор JSON.Net вместо стандартного DataContractSerializer?
- Исключение «Исключительная привязка к реляционной ссылке» с JSON.Net
- Отключить classы JSON до C #
- Настройка пользовательских дат WebApi Json.NET
- Как создать JSON.NET Date to String custom Converter
- JSON.NET Parser * кажется * будет двойной сериализацией моих объектов
Я считаю, что единственный способ – сделать JsonConverter для всего Dictionary
или Dictionary
.
Словарные ключи не считаются значениями и не будут запускаться через JsonConverters. TypeConverters были бы решением, но строка по умолчанию для преобразования enums будет введена до того, как она посмотрит на TypeConverters.
Итак … Я не думаю, что это можно сделать по-другому.
РЕДАКТИРОВАТЬ:
Не полностью протестирован, но я использую что-то вроде этого в моем проекте:
public class DictionaryWithSpecialEnumKeyConverter : JsonConverter { public override bool CanWrite { get { return false; } } public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer) { throw new NotSupportedException(); } public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer) { if (reader.TokenType == JsonToken.Null) return null; var valueType = objectType.GetGenericArguments()[1]; var intermediateDictionaryType = typeof(Dictionary<,>).MakeGenericType(typeof(string), valueType); var intermediateDictionary = (IDictionary)Activator.CreateInstance(intermediateDictionaryType); serializer.Populate(reader, intermediateDictionary); var finalDictionary = (IDictionary)Activator.CreateInstance(objectType); foreach (DictionaryEntry pair in intermediateDictionary) finalDictionary.Add(Enum.Parse(MyEnum, "Code" + pair.Key, false), pair.Value); return finalDictionary; } public override bool CanConvert(Type objectType) { return objectType.IsA(typeof(IDictionary<,>)) && objectType.GetGenericArguments()[0].IsA(); } }
Вам понадобится этот маленький помощник:
public static bool IsA(this Type type, Type typeToBe) { if (!typeToBe.IsGenericTypeDefinition) return typeToBe.IsAssignableFrom(type); var toCheckTypes = new List { type }; if (typeToBe.IsInterface) toCheckTypes.AddRange(type.GetInterfaces()); var basedOn = type; while (basedOn.BaseType != null) { toCheckTypes.Add(basedOn.BaseType); basedOn = basedOn.BaseType; } return toCheckTypes.Any(x => x.IsGenericType && x.GetGenericTypeDefinition() == typeToBe); }
Надеюсь, что это работает для вас.
Вот общее решение проблемы использования json с Dictionary с любым типом ключа:
[JsonObject] public class MyKeyValuePair { public TKey MyKey; public TValue MyValue; [JsonConstructor] public MyKeyValuePair() { } public MyKeyValuePair(TKey t1, TValue t2) { MyKey = t1; MyValue = t2; } } [JsonObject] public class MyDictionaty { public ICollection> Collection; [JsonConstructor] public MyDictionaty() { } public MyDictionaty(Dictionary refund) { Collection = BuildMyKeyValuePairCollection(refund); } internal Dictionary ToDictionary() { return Collection.ToDictionary(pair => pair.MyKey, pair => pair.MyValue); } private ICollection> BuildMyKeyValuePairCollection(Dictionary refund) { return refund.Select(o => new MyKeyValuePair(o.Key, o.Value)).ToList(); } } [JsonObject] public class ClassWithDictionary { [JsonProperty] private readonly MyDictionary _myDictionary; private Dictionary _dic; [JsonConstructor] public ClassWithDictionary() { } public ClassWithDictionary(Dictionary dic) { _dic= dic; _myDictionary = new MyDictionaty(dic); } public Dictionary GetTheDictionary() { _dic = _dic??_myDictionary.ToDictionary(); return _dic; } }