可以使用XmlSerializer直接序列化和反序列化xml
注意其中几个关键点
1. 读取xml文件时,以独占方式,这样可以避免文件被别的程序打开时读取报错的问题。
using (FileStream fileStream = new FileStream(filePath, FileMode.Open, FileAccess.Read, FileShare.ReadWrite)) {}
2. 在序列化反序列化时,当类型DeviceInfoDto
对应的xml中元素的名字不是DeviceInfoDto
而是device
时,有两种解决方法:
方法一:
用该特性[XmlArrayItem("device")]
指定类型DeviceInfoDto
对应的在xml中的元素的名字device。
[XmlArrayItem("device")] public List<DeviceInfoDto> devices { get; set; } = new();
方法二:
需要变通:改以下代码中List<DeviceInfoDto>
为List<device>
其中device
为一个继承自DeviceInfoDto
类型的子类
public List<device> devices { get; set; } = new();
public class device:DeviceInfoDto { }
3. 反序列化时对应C#类型DateTime时的报错处理
当直接使用一个类型为Datetime属性名为creationDate来在反序列化xml文件中的日期时间类型的接收者时,反序列化会报错:【字符串“2020-07-31 09:29:16”不是有效的AllXsd值】。于是采入引入一个中间属性stringCreationDate(类型为string)和存储其值的字段_creationDate(类型为DateTime)的方式,运行过程1.当反序列化时从xml文件读取元素名为creationDate的元素值并通过stringCreationDate的set访问器将其值在转值为DateTime类型后存放于字段_creationDate中 2.当C#代码中其他地方要访问这个值时,就从creationDate属性的get访问器传递出_creationDate字段的值。缺点:需引入不必要的属性和字段,而且属性名还必须是public的,这会让外部访问者对这个中间属性感到疑惑。
private DateTime _creationDate; [XmlElement(ElementName = "creationDate")] public string stringCreationDate { get => _creationDate.ToString("yyyy-MM-dd HH:mm:ss"); set => _creationDate = Convert.ToDateTime(value); } [XmlIgnore] public DateTime creationDate { get => _creationDate; }
4. 序列化反序列化时,XmlSerializer可识别的常用的C#特性标签 Attribute
a. [XmlRoot("udid")]
用于C#类名上,用于指明根类型对应的xml中的元素名
b. [XmlElement("creationDate")]
用于C#属性或字段上,用于指明属性对应的xml元素名或者叫标签名为 creationDate
c. [XmlAttribute("frequency")]
用于C#属性或字段上,用于指明属性或字段对应的XML元素的属性名为 frequency
d. [XmlIgnore]
用于C#属性或字段上,用来标记该字段在序列化反序列化XML文档时是要忽略的C#属性或字段
e. [XmlArrayItem("device")]
用于C# List<T>
集合类型的属性或字段上,用来标记集合中的元素类型T
在XML文档中对应的元素名称