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 соответственно.

У меня есть собственный конвертер, который может обрабатывать преобразование. Но как я могу указать JsonConverter для использования с ключами словаря?

Я считаю, что единственный способ – сделать 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; } } 
  • Как использовать JSON.NET для десериализации в вложенный / рекурсивный словарь и список?
  • Web API 2: как вернуть JSON с именами свойств camelCased, на объекты и их под-объекты
  • Ошибка при запуске JSON.NET при сериализации Mongo ObjectId
  • Не удалось загрузить файл или сборку «Newtonsoft.Json, Version = 4.5.0.0, Culture = neutral, PublicKeyToken = 30ad4fe6b2a6aeed»
  • Как написать Json-файл в C #?
  • Использование пользовательской десериализации тела WCF без изменения десериализации шаблона URI
  • JSON.net сериализуется непосредственно из oledbconnection
  • Как сообщить Json.Net глобально применить StringEnumConverter ко всем enumsм
  • Ошибка десериализации Newtonsoft JSON.net, где поля в порядке изменения JSON
  • Как я могу разбирать JSON с C #?
  • Как обрабатывать как отдельный элемент, так и массив для одного и того же свойства с помощью JSON.net
  • Давайте будем гением компьютера.