¿Puede Newtonsoft deserializar correctamente una lista de objetos heredados o de subclase? [duplicar]

Dec 08 2020

Usando Newtonsoft (con TypeNameHandling.All), si serializo un List<Animal>que incluye a <Cat>y a <Dog>, cuando deserializo la cadena de nuevo en a List<Animal>, pierden su herencia y se vuelven justos <Animals>.

¿Hay alguna manera de que Newtonsoft conserve las subclases a medida que se deserializan de nuevo en una Listclase principal?

Aquí están los 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
    {

    }
}

Aquí está la prueba:

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);
        }

Así es como se ve la lista original de Zoo.Animal (observe los tipos de perros y gatos):

Ahora, una vez serializado, podemos ver que las subclases están incluidas correctamente. Debería ser suficiente información para deserializarlo nuevamente en un objeto similar:

{
    "$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"
            }
        ]
    }
}

Pero cuando se deserializa, Dogy Catdesaparecen. Son solo Animalsahora:

¿Hay alguna forma de que Newtonsoft deserialice correctamente el JSON y mantenga los tipos de subclase?

Muchas gracias por los consejos.

Respuestas

1 Richard Dec 08 2020 at 12:10

Necesita agregar configuraciones tanto en el serializador como en el deserializador.

prueba esto:

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);
}

Debería imprimir: Perro Gato