gpt4 book ai didi

c# - XmlSerializer 和 xsi : deserialization

转载 作者:太空宇宙 更新时间:2023-11-03 15:07:13 25 4
gpt4 key购买 nike

由于 xsi:type="p:OUTPUT-HEADER"属性,我很难尝试反序列化对应于 WCF SOAP 服务故障详细信息部分的这段 XML 代码:

<p:OUTPUT-HEADER xsi:type="p:OUTPUT-HEADER" xmlns:p="http://aaa.bbb.ccc/v2" xmlns:ns0="http://aaa.bbb.ccc/v2" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
<FAULT>
<p:COD-ERROR>2951</p:COD-ERROR>
<p:COD-SEV>8</p:COD-SEV>
<p:MSG-ERROR>Error message</p:MSG-ERROR>
</FAULT>
<CNL-OUT>xxx</CNL-OUT>
</p:OUTPUT-HEADER>

这些是我正在使用的类:

[XmlInclude(typeof(OutputHeader))]
public abstract class FaultDetail
{
[XmlElement(ElementName = "FAULT", Namespace = "")]
public Fault FaultSection{ get; set; }

[XmlElement(ElementName = "CNL-OUT", Namespace = "")]
public string ClnOut{ get; set; }
}

[XmlRoot(ElementName = "OUTPUT-HEADER", Namespace = "http://aaa.bbb.ccc/v2")]
public class OutputHeader : FaultDetail
{
}

public class Fault
{
[XmlElement(ElementName = "COD-ERROR")]
public int CodigoError { get; set; }

[XmlElement(ElementName = "COD-SEV")]
public int Severidad { get; set; }

[XmlElement(ElementName = "MSG-ERROR")]
public string Mensaje { get; set; }

}

XmlSerializer:

XmlSerializer x = new XmlSerializer(typeof(OutputHeader));

调用反序列化方法时出现的错误:

“无法识别指定的类型:名称='OUTPUT-HEADER',命名空间='http://aaa.bbb.ccc/v2 ', 在 http://aaa.bbb.ccc/v2 '>。”

是否可以修饰类以正确反序列化此 XML?非常感谢任何想法,谢谢!

最佳答案

而不是 XmlSerializer,您似乎必须使用 DataContractSerializer反序列化此 XML。这个序列化器是 default serializer for WCF因此您只需要删除指定使用 XmlSerializer 的代码。

按如下方式设计您的类型:

[DataContract(Namespace = "")]
public abstract class OutputHeaderBase
{
[DataMember(Name = "FAULT", Order = 1)]
public Fault FaultSection { get; set; }

[DataMember(Name = "CNL-OUT", Order = 2)]
public string ClnOut { get; set; }
}

[DataContract(Name = "OUTPUT-HEADER", Namespace = "http://aaa.bbb.ccc/v2")]
public class OutputHeader : OutputHeaderBase
{
}

[DataContract(Name = "FAULT", Namespace = "http://aaa.bbb.ccc/v2")]
public class Fault
{
[DataMember(Name = "COD-ERROR", Order = 1)]
public int CodigoError { get; set; }

[DataMember(Name = "COD-SEV", Order = 2)]
public int Severidad { get; set; }

[DataMember(Name = "MSG-ERROR", Order = 3)]
public string Mensaje { get; set; }
}

然后将您的操作合约声明为返回(或接受)OutputHeader 类型的对象(不是OutputHeaderBase)。

最后通过删除 [XmlSerializerFormat] 切换回数据契约(Contract)序列化从您的服务和/或运营契约(Contract)中,您应该准备就绪。有关切换的详细信息,请参阅 Using the XmlSerializer Class

(另请注意,Fault 的属性需要放入正确的命名空间中。)

为什么这样行得通

"xsi:type" attribute 是一个 w3c 标准属性,允许元素显式断言其类型。两者 XmlSerializerDataContractSerializer在序列化多态类型时使用此属性来传达实际类型信息。但是,以下元素:

<p:OUTPUT-HEADER 
xsi:type="p:OUTPUT-HEADER"
xmlns:p="http://aaa.bbb.ccc/v2"
xmlns:ns0="http://aaa.bbb.ccc/v2"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" >
</p:OUTPUT-HEADER>

在命名空间 http://aaa.bbb.ccc/v2 中有基本类型 OUTPUT-HEADER 和命名空间中的子类型 OUTPUT-HEADER http://aaa.bbb.ccc/v2 -- 即类型和子类型信息相同,因此 xsi:type 属性是多余的。

但是,如果它是多余的,它应该是无害的,对吧?您可以为 XmlSerializer 设计一个类型层次结构,如下所示:

[XmlRoot(ElementName = "OUTPUT-HEADER", Namespace = "http://aaa.bbb.ccc/v2")]
[XmlInclude(typeof(OutputHeaderSubclass))] // Artificial subtype to trigger handling of the `xsi:type` attribute.
[XmlInclude(typeof(OutputHeader))]
public class OutputHeader
{
[XmlElement(ElementName = "FAULT", Namespace = "")]
public Fault FaultSection { get; set; }

[XmlElement(ElementName = "CNL-OUT", Namespace = "")]
public string ClnOut { get; set; }
}

[XmlRoot(ElementName = "OUTPUT-HEADER-SUBCLASS", Namespace = "http://aaa.bbb.ccc/v2")]
public class OutputHeaderSubclass : OutputHeader
{
}

然后反序列化为 OutputHeader 可能会很好地工作。不幸的是,事实并非如此。 XmlSerializer 为冗余属性抛出异常而不是处理它。相反,DataContractSerializer 没有,所以要使用它。

关于c# - XmlSerializer 和 xsi : deserialization,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/42814684/

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