gpt4 book ai didi

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

转载 作者:数据小太阳 更新时间:2023-10-29 02:08:11 24 4
gpt4 key购买 nike

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

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

我有这些对象的字典:

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/>

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

更新。

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

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 类,不如使用 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>

关于c# - 如何使用 XElement 在 C# 中使用 List 序列化对象?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/30561570/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com