如果您真的必须避免“更改类结构”,那么(在我看来)属性不会获得您想要的 XML 结构并帮助您获得可读、可维护的代码。
IXmlSerializable接口允许您手动控制类的序列化/反序列化。实现此接口时,当您的类被XmlSerializer使用时,您可以完全控制创建所需的任何 XML 文本。
下面是一个编译控制台应用程序示例供您使用。它展示了如何在您的SomeClass
XML 中读取和写入自定义节点,以及如何将您的类属性放入这些节点上的 XML 属性中。请注意,我曾经nameof
获得一个属性名称,但如果您愿意,您也可以轻松地将 XML 属性名称硬编码"Attribute1"
为"Attribute2"
。
正如您将看到的,编写比使用属性更麻烦。但是一旦你掌握了它,它就会变得非常简单。它绝对不会改变类结构。
using System.IO;
using System.Xml;
using System.Xml.Schema;
using System.Xml.Serialization;
namespace ConsoleApp1
{
public class SomeClass : IXmlSerializable
{
// element 1
public int prop1 { get; set; }
public int prop2 { get; set; }
// element 2
public int prop3 { get; set; }
public int prop4 { get; set; }
public int prop5 { get; set; }
public int prop6 { get; set; }
#region IXmlSerializable
public XmlSchema GetSchema()
{
return null;
}
public void WriteXml(XmlWriter writer)
{
writer.WriteStartElement("Element1");
writer.WriteAttributeString(nameof(prop1), prop1.ToString());
writer.WriteAttributeString(nameof(prop2), prop2.ToString());
writer.WriteEndElement();
writer.WriteStartElement("Element2");
writer.WriteAttributeString(nameof(prop3), prop3.ToString());
writer.WriteAttributeString(nameof(prop4), prop4.ToString());
writer.WriteAttributeString(nameof(prop5), prop5.ToString());
writer.WriteAttributeString(nameof(prop6), prop6.ToString());
writer.WriteEndElement();
}
public void ReadXml(XmlReader reader)
{
// element 1
reader.Read();
reader.MoveToAttribute(nameof(prop1));
if (reader.ReadAttributeValue())
{
prop1 = int.Parse(reader.Value);
}
reader.MoveToAttribute(nameof(prop2));
if (reader.ReadAttributeValue())
{
prop2 = int.Parse(reader.Value);
}
// element 2
reader.Read();
reader.MoveToAttribute(nameof(prop3));
if (reader.ReadAttributeValue())
{
prop3 = int.Parse(reader.Value);
}
reader.MoveToAttribute(nameof(prop4));
if (reader.ReadAttributeValue())
{
prop4 = int.Parse(reader.Value);
}
reader.MoveToAttribute(nameof(prop5));
if (reader.ReadAttributeValue())
{
prop5 = int.Parse(reader.Value);
}
reader.MoveToAttribute(nameof(prop6));
if (reader.ReadAttributeValue())
{
prop6 = int.Parse(reader.Value);
}
}
#endregion
}
class Program
{
static void Main()
{
string tempPath = "c:\\temp\\test.xml";
XmlSerializer serializer = new XmlSerializer(typeof(SomeClass));
// build an instance to serialize
SomeClass s1 = new SomeClass
{
prop1 = 1,
prop2 = 2,
prop3 = 3,
prop4 = 4,
prop5 = 5,
prop6 = 6
};
// serialize it
using (StreamWriter sw = new StreamWriter(tempPath))
{
serializer.Serialize(sw, s1);
}
/* Produces the following XML:
<?xml version="1.0" encoding="utf-8"?>
<SomeClass>
<Element1 prop1="1" prop2="2" />
<Element2 prop3="3" prop4="4" prop5="5" prop6="6" />
</SomeClass>
*/
// deserialize
SomeClass s2;
using (StreamReader sr = new StreamReader(tempPath))
{
s2 = (SomeClass)serializer.Deserialize(sr);
}
// check contents of s2 as you please
}
}
}
(如果你喜欢这个,你应该在实际部署它之前清理它 - 没有错误处理存在,例如 on int.Parse
。它只是为了说明在IXmlSerializable
不改变类结构的情况下的使用。)