- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我正在尝试序列化一个继承自实现 IXmlSerializable 的基类的类。
名为 PropertyBag 的基类是一个允许动态属性的类 (credits to Marc Gravell)。
我实现了 IXmlSerializable,以便将动态属性(存储在字典中)编写为普通的 xml 元素。
例如在序列化具有公共(public)属性(非动态)Name 和动态属性 Age 的类 Person 时,我希望它生成以下 XML:
<Person>
<Name>Tim</Name>
<DynamicProperties>
<Country>
<string>USA</string>
</Country>
</DynamicProperties>
<Person>
我可以获得该部件以在基类 PropertyBag 中使用以下 WriteXml 实现:
public void WriteXml(System.Xml.XmlWriter writer)
{
writer.WriteStartElement("DynamicProperties");
// serialize every dynamic property and add it to the parent writer
foreach (KeyValuePair<string, object> kvp in properties)
{
writer.WriteStartElement(kvp.Key);
StringBuilder itemXml = new StringBuilder();
using (XmlWriter itemWriter = XmlWriter.Create(itemXml))
{
// serialize the item
XmlSerializer xmlSer = new XmlSerializer(kvp.Value.GetType());
xmlSer.Serialize(itemWriter, kvp.Value);
// read in the serialized xml
XmlDocument doc = new XmlDocument();
doc.LoadXml(itemXml.ToString());
// write to modified content to the parent writer
writer.WriteRaw(doc.DocumentElement.OuterXml);
}
writer.WriteEndElement();
}
writer.WriteEndElement();
}
但是,在序列化 Person 类时,它不再序列化普通(非动态)属性,除非我覆盖 Person 中的 WriteXml 方法(我不想这样做)。有什么办法可以在基类中自动添加静态属性吗?我知道我可以使用反射手动执行此操作,但我想知道 .Net Framework 中是否有一些内置功能?
最佳答案
我在 XmlSerializer
(以及其他各种序列化 API)上花费了很多时间,我很确定这很简单:你做不到。实现 IXmlSerializable
要么全有,要么全无。
我能想到的最接近的是作弊并将所有固定属性移动到一个子对象;这会给你稍微不同的 xml - 类似于:
<FixedProperties>
<Name>Tim</Name>
</FixedProperties>
<DynamicProperties>
<Country>
<string>USA</string>
</Country>
</DynamicProperties>
但我希望它会起作用。您将在基础对象上具有直通属性:
[Browsable(false), EditorBrowsable(EditorBrowsableState.Never)]
public FixedProperties FixedProps {get;set;}
public string Name {
get {return FixedProps.Name;}
set {FixedProps.Name = value;}
}
有道理吗?您也可以将 Name
标记为 [XmlIgnore]
,但这看起来很多余。在您定制的序列化方法中,您将使用 new XmlSerializer(typeof(FixedProperties))
编辑:这是一个有效的“序列化”示例:
using System;
using System.ComponentModel;
using System.Xml.Serialization;
static class Program
{
static void Main()
{
MyType obj = new MyType { Name = "Fred" };
var ser = new XmlSerializer(obj.GetType());
ser.Serialize(Console.Out, obj);
}
}
public class MyType : IXmlSerializable
{
public MyType()
{
FixedProperties = new MyTypeFixedProperties();
}
[Browsable(false), EditorBrowsable(EditorBrowsableState.Never)]
public MyTypeFixedProperties FixedProperties { get; set; }
[XmlIgnore]
public string Name
{
get { return FixedProperties.Name; }
set { FixedProperties.Name = value; }
}
System.Xml.Schema.XmlSchema IXmlSerializable.GetSchema()
{
return null;
}
void IXmlSerializable.ReadXml(System.Xml.XmlReader reader)
{
throw new System.NotImplementedException();
}
void IXmlSerializable.WriteXml(System.Xml.XmlWriter writer)
{
writer.WriteStartElement("DynamicProperties");
writer.WriteElementString("Foo", "Bar");
writer.WriteEndElement();
fixedPropsSerializer.Serialize(writer, FixedProperties);
}
static readonly XmlSerializer fixedPropsSerializer
= new XmlSerializer(typeof(MyTypeFixedProperties));
}
[XmlRoot("FixedProperties")]
public class MyTypeFixedProperties
{
public string Name { get; set; }
}
关于c# - 在基类中实现 IXmlSerializable 时如何恢复到 'default' XML 序列化?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/1982916/
我需要能够对内部类进行 Xml 序列化,因此我必须实现 IXmlSerializable。 这个类有两个字符串和一个列表。 我知道使用 WriteElementString 和 ReadElement
我有一个看起来像这样的 xml 文件: value value value value 我有两个对象代表这个 xml: class
.NET 中的 XML 序列化通过 XmlSerializer 构造函数的 extraTypes[] 参数允许多态对象。它还允许为实现 IXmlSerializable 的类型自定义 XML 序列化。
我正在实现 IXmlSerializable对于我的类的自定义(反)序列化逻辑,但希望根据 XSD 模式检查 XML 的读写。我添加了 XmlSchemaProviderAttribute : [Xm
假设我有两个类,一个基类和一个派生类。它们是相当简单的类,主要充当数据结构(派生类显然稍微复杂一些)。 public class BaseUserSession { // ... Variou
我试图创建一个通用的 Dictionary实现 IXmlSerializable (归功于 Charles Feduke )。 这是我的试验: Sub Main() Dim z As New
我正在构建一个将数据公开为 XML 的 REST API。我的域层中有一大堆域类,供 API 后面的服务层和我们将提供给客户的客户端 API 使用。 (客户确实可以选择直接与 REST API 交互,
是否有任何标记为 IXmlSerializable 的 .NET 通用集合?我已经尝试过 List 和 Collection,但都没有开箱即用。 在我推出自己的 collection、list 或 d
我对 xml 序列化有一个奇怪的要求。 请引用以下 C# 代码(由于变量 'rootName' 超出范围而无法编译)。我的意图是让我的类(class) GeneralData 成为“一般”。这意味着该
我有一个类: [Serializable] public class Profile { [XmlAttribute] private string[] permissions;
我想对我的对象 Exception 的实例进行 XML 序列化,并将其存储在另一个对象 ExceptionReport 的 XMLNode[] Nodes 属性中。 [System.Diagnosti
我有一个实现 IXmlSerializable 的类,如下所示: public class SomeClass : IXmlSerializable { public SomeSerializ
一旦程序员决定实现IXmlSerializable,实现它的规则和最佳实践是什么?我听说 GetSchema() 应该返回 null 并且 ReadXml 应该在返回之前移动到下一个元素。这是真的?那
我有这样的 xml: 2 6,501698000000 8,414278000000 9,29
对于要使用的正确装饰器或可能需要的任何设计有点困惑。当序列化一个实现 IXmlSerializable 的类时,有没有办法在 XmlRoot 元素中包含命名空间及其前缀? 例如类定义。 using S
我有以下类定义 [XmlRoot(ElementName = "person",Namespace = "MyNamespace")] public class Person : IXmlSerial
我有一个自定义类型的集合属性,它继承自 BindingList。目前,此属性通过 XmlSerializer 进行序列化,即使它没有 Setter。我现在正尝试在此自定义集合类上实现 IXmlSeri
初始问题 我要serialize一个List更改 IXmlSerializable 的 XmlType动态分类(所以我不能使用属性标签来做到这一点) 我已经尝试使用 XmlAttributeOverr
我发现 XmlSerializer 有一些有趣的行为。 如果我尝试序列化一个具有 Type 对象属性并且该对象实现了 IXmlSerializable 的类,序列化程序将抛出一个 InvalidOpe
基本上,最初的问题是我需要将 bool 值序列化为 0 或 1。我找到的解决方案是实现 IXmlSerializable,我确实这样做了。不幸的是,我试图序列化的类是从模式生成的代码,并且上面有一个
我是一名优秀的程序员,十分优秀!