如何在c#中使用XElement序列化对象和列表

本文关键字:序列化 XElement 对象 列表 | 更新日期: 2023-09-27 17:52:16

我有一个包含不同类型成员的对象,像这样:

public class MyObject
{
    public string Str1 = string.Empty;
    public MyEnums.Enum1 E1 = MyEnums.Enum1.Unknown;
    public bool Done = false;
};

我有这些对象的Dictionary

Dictionary<string, MyObject> MyObjectsDic = new Dictionary<string, MyObject>();

和这样的序列化器:

public static void ToXml(string file, string collectionName, Dictionary<string, object> collection)
{
    XElement root = new XElement(collectionName);
    root.Add(collection.Select(x => new XElement("Item", new XAttribute("Object", x.Key),
            x.Value.GetType().GetFields().Select(f => new XElement(f.Name, f.GetValue(x.Value))))));
    root.Save(file);
}

序列化器将抽象Dictionary作为参数,我需要手动转换我的MyObjectsDic。可能我弄错了。

ToXml("MyFile.xml", "MyObjects", MyObjectsDic.ToDictionary(p => p.Key, p => (object)p.Value));

我使用这个建议来创建序列化器。它工作得很好,但我需要添加到MyObject的新成员

List<MyEnums.Enum2> Codes = new List<MyEnums.Enum2>();

并在这里存储一些值

var a = new MyObject {...};
a.Codes.Add(MyEnums.Enum2.Code1);
a.Codes.Add(MyEnums.Enum2.Code2);
MyObjectsDic.Add("Obj1", a);

但是这些列表被序列化成像

这样的文件
<Codes>Code1Code2<Codes/>

不带空格或分隔符。而且我不知道如何在不修改序列化器和不添加新的奇怪代码的情况下使其更具可读性。我得到的唯一想法是在MyObject中保留已经准备好的字符串,而不是列表<…>。它并不优雅,但简单有效。我不读取这些数据,只是写入并保存为日志文件。
或者我需要改变我的序列化器?

乌利希期刊指南。

我使用下面的解决方案,但我收到一个异常在Windows XP。在其他操作系统上运行良好。我修改了代码,使其像helper而不是类扩展。

Exception during dumping MyObjectsDic: There was an error reflecting type 'MyObject'.  
at System.Xml.Serialization.XmlReflectionImporter.ImportTypeMapping(TypeModel model, String ns, ImportContext context, String dataType, XmlAttributes a, Boolean repeats, Boolean openModel, RecursionLimiter limiter)
at System.Xml.Serialization.XmlReflectionImporter.ImportElement(TypeModel model, XmlRootAttribute root, String defaultNamespace, RecursionLimiter limiter)
at System.Xml.Serialization.XmlReflectionImporter.ImportTypeMapping(Type type, XmlRootAttribute root, String defaultNamespace)
at System.Xml.Serialization.XmlSerializer..ctor(Type type, String defaultNamespace)
at System.Xml.Serialization.XmlSerializer..ctor(Type type)
at MyXmlSerializerHelper.SerializeToXElement[T](T obj, XmlSerializer serializer, Boolean omitStandardNamespaces) in MyXmlSerializerHelper.cs:line 16
at MyXmlSerializerHelper.  <SerializeToFile>b__0[T](KeyValuePair'2 x) in MyXmlSerializerHelper.cs:line 5

我唯一的想法是——不同版本的框架或XP上的其他宗教问题…不幸的是,我不能在生产中安装任何其他软件或。net版本。

如何在c#中使用XElement序列化对象和列表

而不是尝试使用反射来手动序列化您的MyObject类,您可以使用XmlSerializer使用以下方法将字典值直接序列化到XElement,然后将结果包含在您正在构建的元素树中:

public static class XObjectExtensions
{
    public static XElement SerializeToXElement<T>(this IDictionary<string, T> collection, string collectionName)
    {
        return new XElement(collectionName, collection.Select(x => new XElement("Item", new XAttribute("Object", x.Key), x.Value.SerializeToXElement().Elements())));
    }
    public static XElement SerializeToXElement<T>(this T obj, XmlSerializer serializer = null, bool omitStandardNamespaces = true)
    {
        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 ?? new XmlSerializer(obj.GetType())).Serialize(writer, obj, ns);
        }
        var element = doc.Root;
        if (element != null)
            element.Remove();
        return element;
    }
}

这将自动导致MyObject的所有字段和属性被正确序列化。使用此方法,生成的XML看起来像:

<MyObjects>
  <Item Object="Obj1">
    <Str1>Test object</Str1>
    <E1>Unknown</E1>
    <Done>false</Done>
    <Codes>
      <Enum2>Code1</Enum2>
      <Enum2>Code2</Enum2>
    </Codes>
  </Item>
</MyObjects>