Changing type of element in XML Serialization

前端 未结 2 945
生来不讨喜
生来不讨喜 2021-01-14 01:15

I am having huge problems with XML serialization. I have two classes, both need to be serializeable. In the inherited class, I would like to change the serialization behavio

2条回答
  •  失恋的感觉
    2021-01-14 02:11

    The reason you get the error is that, during XmlSerializer code generation, the code generator doesn't understand that the two potential NAME elements on Cat will never be simultaneously serialized, so throws the exception.

    Instead, you can apply XmlAnyElementAttribute to a virtual property returning an XElement, then manually create and return an appropriate XElement for the name for each class in the hierarchy:

    [XmlInclude(typeof(Cat))]
    public class Animal
    {
        [XmlIgnore]
        public string Name { get; set; }
    
        [XmlAnyElement]
        public virtual XElement XmlName
        {
            get
            {
                return Name == null ? null : new XElement("NAME", Name);
            }
            set
            {
                Name = (value == null ? null : value.Value);
            }
        }
    }
    
    public class Cat : Animal
    {
        // Must be cached as per https://msdn.microsoft.com/en-us/library/system.xml.serialization.xmlserializer%28v=vs.110%29.aspx
        static XmlSerializer nameSerializer;
    
        static Cat()
        {
            nameSerializer = new XmlSerializer(typeof(NameAndType), new XmlRootAttribute("NAME"));
        }
    
        [XmlIgnore]
        public NameAndType Name2 { get; set; }
    
        [XmlAnyElement]
        public override XElement XmlName
        {
            get
            {
                return (Name2 == null ? null : XObjectExtensions.SerializeToXElement(Name2, nameSerializer, true));
            }
            set
            {
                Name2 = (value == null ? null : XObjectExtensions.Deserialize(value, nameSerializer));
            }
        }
    }
    

    Using the extension methods:

    public static class XObjectExtensions
    {
        public static T Deserialize(this XContainer element)
        {
            return element.Deserialize(new XmlSerializer(typeof(T)));
        }
    
        public static T Deserialize(this XContainer element, XmlSerializer serializer)
        {
            using (var reader = element.CreateReader())
            {
                object result = serializer.Deserialize(reader);
                if (result is T)
                    return (T)result;
            }
            return default(T);
        }
    
        public static XElement SerializeToXElement(this T obj)
        {
            return obj.SerializeToXElement(new XmlSerializer(obj.GetType()), true);
        }
    
        public static XElement SerializeToXElement(this T obj, XmlSerializer serializer, bool omitStandardNamespaces)
        {
            var doc = new XDocument();
            using (var writer = doc.CreateWriter())
            {
                XmlSerializerNamespaces ns = null;
                if (omitStandardNamespaces)
                    (ns = new XmlSerializerNamespaces()).Add("", ""); // Disable the xmlns:xsi and xmlns:xsd lines.
                serializer.Serialize(writer, obj, ns);
            }
            var element = doc.Root;
            if (element != null)
                element.Remove();
            return element;
        }
    }
    

    Which, for a List, produces XML like this:

    
        
            duck
        
        
            
                Smokey
                Siamese
            
        
    
    

提交回复
热议问题