A Newtonsoft pode desserializar corretamente uma lista de objetos herdados ou de subclasse? [duplicado]

Dec 08 2020

Usando Newtonsoft (with TypeNameHandling.All), se eu serializar a List<Animal>que inclui a <Cat>e a <Dog>, quando desserializo a string de volta em a List<Animal>, eles perdem sua herança e se tornam justos <Animals>.

Existe uma maneira de a Newtonsoft preservar as subclasses à medida que são desserializadas de volta para uma Listde sua classe pai?

Aqui estão os objetos:

 public class Zoo
    {
        public List<Animal> Animals;

        public Zoo()
        {
            Animals = new List<Animal>();
        }
    }
    public class Animal
    {

    }

    public class Dog : Animal
    {

    }

    public class Cat : Animal
    {

    }
}

Aqui está o teste:

static void Main(string[] args)
        {
            var zoo = new Zoo();
            var dog = new Dog();
            var cat = new Cat();
            zoo.Animals.Add(dog);
            zoo.Animals.Add(cat);

            var json = JsonConvert.SerializeObject(zoo, Formatting.None,
                new JsonSerializerSettings
                {
                    TypeNameHandling = TypeNameHandling.All
                });

            var newZoo = JsonConvert.DeserializeObject<Zoo>(json);
        }

Esta é a aparência da lista Zoo.Animal original (observe os tipos Cão e Gato):

Agora, uma vez serializado, podemos ver que as subclasses estão incluídas corretamente. Deve haver informações suficientes para desserializá-lo de volta em um objeto semelhante:

{
    "$type": "Sample.Sandbox.Zoo, Sample.Sandbox", "Animals": { "$type": "System.Collections.Generic.List`1[[Sample.Sandbox.Animal, Sample.Sandbox]], System.Private.CoreLib",
        "$values": [ { "$type": "Sample.Sandbox.Dog, Sample.Sandbox"
            },
            {
                "$type": "Sample.Sandbox.Cat, Sample.Sandbox"
            }
        ]
    }
}

Mas quando ele é desserializado, o Doge Catdesaparece. Eles são Animalsagora:

Existe uma maneira de a Newtonsoft desserializar adequadamente o JSON e manter os tipos de subclasse?

Muito obrigado por todas as dicas.

Respostas

1 Richard Dec 08 2020 at 12:10

Você precisa adicionar configurações no serializador e desserializador.

tente isto:

JsonSerializerSettings settings = new JsonSerializerSettings { TypeNameHandling = TypeNameHandling.All };
var json = JsonConvert.SerializeObject(zoo,settings);    
var newZoo = JsonConvert.DeserializeObject<Zoo>(json,settings);
    
foreach (var currentAnimal in newZoo.Animals)
{ 
   Console.WriteLine(currentAnimal.GetType().Name);
}

Deve imprimir: Cão Gato