El XmlSerializer
no puede manejar una interfaz porque no sabe qué tipos crear al deserializar. Para evitar esto, debe manejar esa parte de la serialización implementando la interfaz IXmlSerializable
. Esto le permite grabar el tipo para que pueda volver a crearlo (deserializarlo).
La clase ListOfIAnimal
a continuación muestra cómo heredé y extendí la lista genérica List<IAnimal>
para implementar la interfaz requerida. Comprimí tus viejas clases y agregué un campo extra sin interfaz para cada una de ellas, así pude ver que las clases concretas estaban siendo serializadas y deserializadas apropiadamente.
En comparación con su código, estoy usando el nuevo tipo ListOfIAnimal
en lugar de List<IAnimal>
, los otros cambios son solo un poco de refactorización.
Su código completo, simplemente cópielo en su propio archivo .cs, llame a la primera función para recorrerlo.
using System;
using System.Collections.Generic;
using System.IO;
using System.Text;
using System.Xml;
using System.Xml.Serialization;
namespace Serialiser
{
static class SerialiseInterface
{
public static void SerialiseAnimals()
{
String finalXml;
// Serialize
{
var animals = new ListOfIAnimal{
new Dog() { Age = 5, Teeth = 30 },
new Cat() { Age = 6, Paws = 4 }
};
var xmlSerializer = new XmlSerializer(animals.GetType());
var stringBuilder = new StringBuilder();
var xmlTextWriter = XmlTextWriter.Create(stringBuilder, new XmlWriterSettings { NewLineChars = "\r\n", Indent = true });
xmlSerializer.Serialize(xmlTextWriter, animals);
finalXml = stringBuilder.ToString();
}
// Deserialise
{
var xmlSerializer = new XmlSerializer(typeof(ListOfIAnimal));
var xmlReader = XmlReader.Create(new StringReader(finalXml));
ListOfIAnimal animals = (ListOfIAnimal)xmlSerializer.Deserialize(xmlReader);
}
}
}
public class ListOfIAnimal : List<IAnimal>, IXmlSerializable
{
public ListOfIAnimal() : base() { }
#region IXmlSerializable
public System.Xml.Schema.XmlSchema GetSchema() { return null; }
public void ReadXml(XmlReader reader)
{
reader.ReadStartElement("ListOfIAnimal");
while (reader.IsStartElement("IAnimal"))
{
Type type = Type.GetType(reader.GetAttribute("AssemblyQualifiedName"));
XmlSerializer serial = new XmlSerializer(type);
reader.ReadStartElement("IAnimal");
this.Add((IAnimal)serial.Deserialize(reader));
reader.ReadEndElement(); //IAnimal
}
reader.ReadEndElement(); //ListOfIAnimal
}
public void WriteXml(XmlWriter writer)
{
foreach (IAnimal animal in this)
{
writer.WriteStartElement("IAnimal");
writer.WriteAttributeString("AssemblyQualifiedName", animal.GetType().AssemblyQualifiedName);
XmlSerializer xmlSerializer = new XmlSerializer(animal.GetType());
xmlSerializer.Serialize(writer, animal);
writer.WriteEndElement();
}
}
#endregion
}
public interface IAnimal { int Age { get; set; } }
public class Dog : IAnimal { public int Age { get; set;} public int Teeth { get; set;} }
public class Cat : IAnimal { public int Age { get; set;} public int Paws { get; set;} }
}
pensé en dejar deserializar como ejercicio para el lector, pero el código would'n ser muy útil sin ella.
Tal vez este tema de ayuda que http: // stackoverflow.com/questions/10225174/using-datacontractserializer-and-dataprotectionprovider-to-serialize-and-encryp – saramgsilva