我有几个具有不同根元素的XML文件,但是相同类型的子元素,我希望能够创建一个单独的类来保存不同的根元素,而另一个类来保存每个子元素。 以下是XML文件外观的两个示例。
文件1:
<?xml version="1.0" encoding="utf-8" ?>
<Sandra>
<Address>
<Street></Street>
<Number></Number>
</Address>
</Sandra>
文件2:
<?xml version="1.0" encoding="utf-8" ?>
<John>
<Address>
<Street></Street>
<Number></Number>
</Address>
</John>
我希望能够使用两个类来序列化和反序列化,例如:
[Serializable]
[XmlRoot]
public class Person
{
[XmlElement("Address")]
public List<Address> Adresses { get; set; }
}
[Serializable]
public class Address
{
public string Street { get; set; }
public string Number { get; set; }
}
我尝试使用以下方式阅读:
var ser = new XmlSerializer(typeof(Person));
var reader = XmlReader.Create("person1.xml");
var person = (Person)ser.Deserialize(reader);
但我得到"There is an error in XML document (2, 2)."
,因为反序列化器期望<"Person">
根元素而不是<"John">
或<"Paul">
。如果我将[XmlRoot]
更改为[XmlRoot("John")]
,它的工作正常,但这正是我在这里要避免的。
此外,我必须能够使用相同的奇怪XML结构将其序列化,因此我需要将根元素名称存储在Person类中。
我知道我可以简单地创建自己的(de)序列化程序,但我想知道是否可以使用现有方法实现它。
编辑1:(回滚)。
编辑2:回滚“编辑1”的修改,因为我找到了一种更简单的方法来实现我所需要的。请参阅下面的答案。
答案 0 :(得分:9)
找到一个简洁快捷的方法来解决我的问题! 我只需在实例化XmlSerializer时使用XmlRootAttribute。这样我就可以在运行时设置根元素的名称。
var personsName = "Sandra";
var ser = new XmlSerializer(typeof(Person),
new XmlRootAttribute { ElementName = personsName });
var reader = XmlReader.Create("person1.xml");
var person = (Person)ser.Deserialize(reader);
当然,如果我想序列化它也会以同样的方式工作。
答案 1 :(得分:3)
使用人名作为XML元素名称感觉有点不稳定。每个元素代表什么,John
或Person
?
如果您能够控制这些XML文件的形状,可能会更好地表示它们:
<?xml version="1.0" encoding="utf-8" ?>
<Person name="Sandra">
<Address>
<Street></Street>
<Number></Number>
</Address>
</Person>
然后您将有一种简单的方法将名称存储在属性中,映射为XML属性:
[Serializable]
[XmlRoot]
public class Person
{
[XmlElement("Address")]
public List<Address> Adresses { get; set; }
[XmlAttribute("name")]
public string Name { get; set;}
}
答案 2 :(得分:1)
在您的类中实现IXmlSerializable,并按照您希望的方式进行序列化:
ReadXml(System.Xml.XmlReader reader)
和WriteXml(System.Xml.XmlWriter writer)
示例:
[Serializable]
public class Threshold : IXmlSerializable
{
public int Type {get;set;}
public object Value {get;set;}
public string Name {get;set;}
public void ReadXml(System.Xml.XmlReader reader)
{
XElement thresholdXML = XElement.Load(reader);
if (!thresholdXML.HasElements || thresholdXML.IsEmpty)
return;
Type = (ThresholdType)int.Parse(thresholdXML.Element("Type").Value);
Value = Type.Equals(ThresholdType.Complex) ? thresholdXML.Element("Value").Value : (object)Decimal.Parse(thresholdXML.Element("Value").Value);
Name = thresholdXML.Element("Name").Value;
}
public System.Xml.Schema.XmlSchema GetSchema()
{
return null;
}
public void WriteXml(System.Xml.XmlWriter writer)
{
XmlSerializerNamespaces xmlnsEmpty = new XmlSerializerNamespaces();
xmlnsEmpty.Add("", "");
writer.WriteElementString("Type", ((int)Type).ToString("D"));
writer.WriteElementString("Value", Value.ToString());
writer.WriteElementString("Name", Name);
}
}
答案 3 :(得分:0)
使用根元素作为ArrayOfClassName,然后尝试。
<ArrayOfAlarmSummary>
<AlarmSummary>
<ClientId>1</ClientId>
<StationId>2</StationId>
<StationName>Station-2</StationName>
<DateTime>13/09/15</DateTime>
<TagName>AI2_2</TagName>
<Description>TR DC Current</Description>
<Units>Amps</Units>
<NormalOperation>10 to 100</NormalOperation>
<AlarmValue>132.48</AlarmValue>
<AlarmDescription>
</AlarmDescription>
</AlarmSummary>
<AlarmSummary>
<ClientId>1</ClientId>
<StationId>2</StationId>
<StationName>Station-2</StationName>
<DateTime>13/09/15</DateTime>
<TagName>AI2_2</TagName>
<Description>TR AC Current</Description>
<Units>Amps</Units>
<NormalOperation>10 to 100</NormalOperation>
<AlarmValue>132.48</AlarmValue>
<AlarmDescription>
</AlarmDescription>
</AlarmSummary>
</ArrayOfAlarmSummary>
这是反序列化类,使用它可以反序列化您的Xml文件。
public class SerializeDeserialize<T>
{
StringBuilder sbData;
StringWriter swWriter;
XmlDocument xDoc;
XmlNodeReader xNodeReader;
XmlSerializer xmlSerializer;
public SerializeDeserialize()
{
sbData = new StringBuilder();
}
public string SerializeData(T data)
{
XmlSerializer employeeSerializer = new XmlSerializer(typeof(T));
swWriter = new StringWriter(sbData);
employeeSerializer.Serialize(swWriter, data);
return sbData.ToString();
}
public T DeserializeData(string dataXML)
{
xDoc = new XmlDocument();
xDoc.LoadXml(dataXML);
xNodeReader = new XmlNodeReader(xDoc.DocumentElement);
xmlSerializer = new XmlSerializer(typeof(T));
var employeeData = xmlSerializer.Deserialize(xNodeReader);
T deserializedEmployee = (T)employeeData;
return deserializedEmployee;
}
}
致电班级
var appDomain = System.IO.Directory.GetCurrentDirectory() + "\\AlarmFiles";
var strPath = Path.Combine(appDomain,
"Alarm_" + DateTime.Now.ToString("ddMMyyyy") + Constants.FileType.XmlFile);
var fi = new FileInfo(strPath);
if (fi.Exists)
{
try
{
var xmlString = System.IO.File.ReadAllText(strPath);
var serializeAlarmSummary =
new SerializeDeserialize<List<AlarmSummary>>();
return serializeAlarmSummary.DeserializeData(xmlString);
}
catch (Exception ex)
{
throw ex;
}
}