Сериализовать json в объект со свойством перехвата всего словаря

Я хотел бы использовать JSON.net для десериализации объекта, но поместить несопоставленные свойства в свойство словаря. Является ли это возможным?

Например, учитывая json,

 {one:1,two:2,three:3}

и класс С#:

public class Mapped {
   public int One {get; set;}
   public int Two {get; set;}
   public Dictionary<string,object> TheRest {get; set;}
}

Может ли JSON.NET десериализоваться в экземпляр со значениями one=1, two=1, TheRest= Dictionary{{"three,3}}


person PhilHoy    schedule 26.07.2011    source источник
comment
Обновил код в моем ответе, чтобы сделать его более общим.   -  person David Hoerster    schedule 26.07.2011


Ответы (2)


Самый простой способ сделать это — использовать атрибут JsonExtensionData для определения универсального словаря.

Пример из документации Json.Net:

public class DirectoryAccount
{
    // normal deserialization
    public string DisplayName { get; set; }

    // these properties are set in OnDeserialized
    public string UserName { get; set; }
    public string Domain { get; set; }

    [JsonExtensionData]
    private IDictionary<string, JToken> _additionalData;

    [OnDeserialized]
    private void OnDeserialized(StreamingContext context)
    {
        // SAMAccountName is not deserialized to any property
        // and so it is added to the extension data dictionary
        string samAccountName = (string)_additionalData["SAMAccountName"];

        Domain = samAccountName.Split('\\')[0];
        UserName = samAccountName.Split('\\')[1];
    }

    public DirectoryAccount()
    {
        _additionalData = new Dictionary<string, JToken>();
    }
}

string json = @"{
  'DisplayName': 'John Smith',
  'SAMAccountName': 'contoso\\johns'
}";

DirectoryAccount account = JsonConvert.DeserializeObject<DirectoryAccount>(json);

Console.WriteLine(account.DisplayName);
// John Smith

Console.WriteLine(account.Domain);
// contoso

Console.WriteLine(account.UserName);
// johns
person Andrew Barrett    schedule 13.05.2016

Вы можете создать CustomCreationConverter, чтобы делать то, что вам нужно. Вот пример (довольно уродливый, но демонстрирующий, как вы можете это сделать):

namespace JsonConverterTest1
{
    public class Mapped
    {
        private Dictionary<string, object> _theRest = new Dictionary<string, object>();
        public int One { get; set; }
        public int Two { get; set; }
        public Dictionary<string, object> TheRest { get { return _theRest; } }
    }

    public class MappedConverter : CustomCreationConverter<Mapped>
    {
        public override Mapped Create(Type objectType)
        {
            return new Mapped();
        }

        public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
        {
            var mappedObj = new Mapped();
            var objProps = objectType.GetProperties().Select(p => p.Name.ToLower()).ToArray();

            //return base.ReadJson(reader, objectType, existingValue, serializer);
            while (reader.Read())
            {
                if (reader.TokenType == JsonToken.PropertyName)
                {
                    string readerValue = reader.Value.ToString().ToLower();
                    if (reader.Read())
                    {
                        if (objProps.Contains(readerValue))
                        {
                            PropertyInfo pi = mappedObj.GetType().GetProperty(readerValue, BindingFlags.IgnoreCase | BindingFlags.Public | BindingFlags.Instance);
                            var convertedValue = Convert.ChangeType(reader.Value, pi.PropertyType);
                            pi.SetValue(mappedObj, convertedValue, null);
                        }
                        else
                        {
                            mappedObj.TheRest.Add(readerValue, reader.Value);
                        }
                    }
                }
            }
            return mappedObj;
        }
    }

    public class Program
    {
        static void Main(string[] args)
        {
            string json = "{'one':1, 'two':2, 'three':3, 'four':4}";

            Mapped mappedObj = JsonConvert.DeserializeObject<Mapped>(json, new MappedConverter());

            Console.WriteLine(mappedObj.TheRest["three"].ToString());
            Console.WriteLine(mappedObj.TheRest["four"].ToString());
        }
    }
}

Таким образом, вывод mappedObj после десериализации строки JSON будет объектом с заполненными свойствами One и Two, а все остальное помещено в файл Dictionary. Конечно, я жестко закодировал значения One и Two как ints, но я думаю, что это демонстрирует, как вы это сделаете.

Надеюсь, это поможет.

EDIT: я обновил код, чтобы сделать его более общим. Я не проверял его полностью, поэтому в некоторых случаях он может дать сбой, но я думаю, что он поможет вам в большинстве случаев.

person David Hoerster    schedule 26.07.2011
comment
Дэвид, это здорово, но я надеялся на более общее решение. - person PhilHoy; 26.07.2011
comment
Да, я как-то быстро собрал. Я не могу вернуться к этому прямо сейчас, но вскоре я сделаю его более общим. Это, вероятно, потребует немного размышлений. Однако базовая структура не изменится — изменится только логика во втором блоке if(reader.Read()). Но, надеюсь, вы понимаете, к чему я клоню. Кстати, очень классный вопрос, который вы задали. - person David Hoerster; 26.07.2011
comment
Обновлен код, чтобы сделать его более универсальным для различных имен и/или типов свойств. - person David Hoerster; 26.07.2011