gpt4 book ai didi

c# - XmlInclude 以(反)序列化单独程序集中的继承类型

转载 作者:太空宇宙 更新时间:2023-11-03 14:28:54 26 4
gpt4 key购买 nike

我有这种情况:

namespace MyApp.Animals
{
public class Dog : MyApp.Categories.Canine
...

所以我正在尝试序列化/反序列化 dog 类。不幸的是,我不能使用 [XmlInclude] 属性,因为通过将它添加到父类 (MyApp.Categories.Canine),我需要添加对父类 (MyApp.Animals) 所在程序集的引用。然而,由于继承,子类已经在引用父类...所以我们有一个循环引用。

是否有实现此目标的最佳实践方法?我可以通过手动序列化/反序列化存在于父类上的属性,然后只对子类进行序列化来解决这个问题,但这不是很优雅。

希望得到一些更好的提示..

更新:添加到示例中以响应下面 John 的评论...

我没有完整的代码示例(我正在使用的代码庞大而复杂),但问题是一些应该反序列化到 xml 或从 xml 序列化的属性包含在父类中,它与子类在不同的程序集中。

将 XmlElement 属性添加到父类不起作用,因为我们实际上是在对子类执行反序列化/序列化,因此它不会迭代到父类。而且我们不能在父级上执行此操作,因为我们无法添加对子级的引用(因为子级已经在引用父级),因此反序列化器将不知道要对哪个子级执行操作。

为了增加额外的复杂性,我遇到问题的对象实际上是通过父对象的以下属性作为更大序列化的一部分进行反序列化/序列化的:

    [XmlElement("ShippingAddress")]
public Location ShippingAddress
{
get { return _shippingAddress; }
set { _shippingAddress = value; }
}

问题是这里的位置类型是子位置类型。因此只有子类型中的属性被反序列化...父类型中的所有属性(也称为 Location,但在不同的命名空间中)都没有。

这样是不是更清楚了?

最佳答案

XmlSerializer 有一个 constructor它采用称为“extraTypes”的类型数组。此数组中的类型将在序列化和反序列化期间可用,就像您添加了 XmlInclude 属性一样。

var serializer = new XmlSerializer(typeof(Canine), new Type[] { typeof(Dog) });

更新:如果您的 Xml 包含 xsi:type 属性,而类型与在根对象上声明的类型不同,则该方法有效。听起来您可能只想将该属性反序列化为不同的类型。在这种情况下,您可以使用 XmlAttributeOverrides替换原始类型的元数据。像这样构建 XmlSerializer:

var overrides = new XmlAttributeOverrides();
var xmlAttributes = new XmlAttributes();
xmlAttributes.XmlElements.Add(new XmlElementAttribute("ShippingAddress", typeof(ITSM.Location)));
overrides.Add(typeof(Order), "Location", xmlAttributes);
var ser = new XmlSerializer(typeof(Order), overrides);

将具有与替换 Order.Location 属性相同的效果:

[XmlElement("ShippingAddress", typeof(ITSM.Location))]

关于c# - XmlInclude 以(反)序列化单独程序集中的继承类型,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/3114870/

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