[英]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));
我用这个建议来制作序列化器。 它运行良好,但我需要添加到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 <...>。 它不优雅,但简单而有效。 我不读这些数据,只是写入并保存为登录文件。
或者我需要改变我那么酷的序列化器?
UPD。
我使用下面的解决方案,但我在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版本。
您可以使用XmlSerializer
使用以下方法将字典值直接序列化为XElement
,然后将结果包含在您正在构建的元素树中,而不是尝试使用反射来手动序列化MyObject
类:
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>
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.