如何使用 XElement 在 C# 中使用 List 序列化对象?

How to serialize object with List inside in C# using XElement?

我反对像这样的不同类型的成员:

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));

我使用 this advice 来制作序列化程序。效果很好,但我需要添加到 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/>

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

更新

我使用了下面的解决方案,但我在 Windows XP 上收到 异常。在其他 OS 上效果很好。我修改了代码,使它像助手而不是 class 扩展。

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 版本。

与其尝试使用反射手动序列化您的 MyObject class,不如使用 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>