【问题标题】:Adding multiple namespaces to MessageContract WCF response object (MessageBodyMember)向 MessageContract WCF 响应对象 (MessageBodyMember) 添加多个命名空间
【发布时间】:2017-02-17 21:45:19
【问题描述】:

我们有一个包含以下合同的 WCF 设置:

    [ServiceContract(
    Namespace = Constants.Namespaces.HL7Namespace,
    Name = Constants.Roles.ContentRequiredDocumentManagementSystem)]
// XmlSerializerFormat is needed to expose the HL7 schema fields without the "Field" suffix on each one, eg: idField
[XmlSerializerFormat]
public interface ICDARequest
{
    [OperationContract(
        // wsdl request action
        Action = Constants.Namespaces.HL7Namespace + ":" + Constants.Interactions.RCMR_IN000029UV01 + "." + Constants.VersionType.NormativeCode + Constants.Version.InteractionVersion,
        // wsdl operation name
        Name = Constants.Interactions.RCMR_IN000029UV01,
        // wsdl response action
        ReplyAction = Constants.Namespaces.HL7Namespace + ":" + Constants.Interactions.RCMR_IN000030UV01 + "." + Constants.VersionType.NormativeCode + Constants.Version.InteractionVersion)]
    SearchMessagesResponse SearchMessages(SearchMessagesRequest RCMR_IN000029UV01);


    [MessageContract(
        IsWrapped = false]
    public class SearchMessagesResponse
    {
        [MessageBodyMember(
            Name = State.Constants.Interactions.RCMR_IN000030UV01,
            Namespace = State.Constants.Namespaces.HL7Namespace)]
        public RCMR_IN000030UV01 data;
    }
}
  • 这些基于使用xsd.exe 的HL7v3 架构生成的类。
  • 然后我们更改了架构以添加自定义元素,使用自定义命名空间来区分它并重新生成类。
  • 这很好。

它补充说:

[System.Xml.Serialization.XmlTypeAttribute(TypeName = "BCCDX.DistributionStatus", Namespace = "urn:bccdx.ca")]
public partial class BCCDXDistributionStatus
{
    [System.Xml.Serialization.XmlElementAttribute("receivedTime", Namespace = "urn:bccdx.ca", IsNullable = false)]
    public TS receivedTime{...}
}

这是我们想要的。

然后在 WCF 服务中我们就可以使用新的类和成员了:

var distStatus = new BCCDXDistributionStatus();
distStatus.receivedTime = CreateTS(locStat.MessageDownloadDate);

然后这会被序列化并通过网络发送出去,如下所示:

<distributionStatus xmlns="urn:bccdx.ca">
    <receivedTime value="201702150956-0800"/>
</distributionStatus>

这几乎是正确的。问题在于 XML 文档没有引用 "urn:bccdx.ca" 命名空间。我假设它会在序列化时自动添加到文档根元素中,但我错了。这就是最终的样子:

<RCMR_IN000030UV01 ITSVersion="XML_1.0" xmlns="urn:hl7-org:v3">
...
</RCMR_IN000030UV01>

当真正想要的是:

<RCMR_IN000030UV01 ITSVersion="XML_1.0" xmlns="urn:hl7-org:v3" xmlns:x="urn:bccdx.ca">
...
</RCMR_IN000030UV01>

注意带有前缀的 urn:bccdx.ca

我想知道,如果我们可以添加多个命名空间,如何通过合同为生成的序列化消息 XML 添加前缀?我在网上看到了覆盖默认序列化程序的提示,但我宁愿不这样做。以前肯定有过这样的想法和处理过吗?

【问题讨论】:

  • 1) 有什么方法可以将其扩展为minimal reproducible example,或者至少可以编译,而不会丢失类型? 2) 你说你使用xsd.exe 来生成你的类,你在某处申请[XmlSerializerFormat] 吗?我没有在问题中看到它。
  • 谢谢,是的,我编辑将XmlSerializerFormat 添加到原始问题中。

标签: c# xml wcf serialization messagecontract


【解决方案1】:

首先,我将假设,在您的服务合同的某处,您通过使用[XmlSerializerFormat] 指定使用XmlSerializer,例如:

[ServiceContract()]
[XmlSerializerFormat]
public interface IService1
{
    [OperationContract(
        // wsdl request action
        Action = Constants.Namespaces.HL7Namespace + ":" + Constants.Interactions.RCMR_IN000029UV01 + "." + Constants.VersionType.NormativeCode + Constants.Version.InteractionVersion,
        // wsdl operation name
        Name = Constants.Interactions.RCMR_IN000029UV01,
        // wsdl response action
        ReplyAction = Constants.Namespaces.HL7Namespace + ":" + Constants.Interactions.RCMR_IN000030UV01 + "." + Constants.VersionType.NormativeCode + Constants.Version.InteractionVersion)]
    SearchMessagesResponse SearchMessages(/* SearchMessagesRequest RCMR_IN000029UV01*/);
}

虽然您的问题中没有提到这一点,但如果您没有这样做,那么您的类型中的 [System.Xml.Serialization.XmlElementAttribute(...)] 属性声明将无效,因为它们会被 DataContractSerializer 忽略。

其次,我将假设您的 RCMR_IN000030UV01 类型当前看起来像这样:

[XmlRoot(ElementName = "RCMR_IN000030UV01", Namespace = "urn:hl7-org:v3")]
public partial class RCMR_IN000030UV01
{
    // The initially auto-generated code
    [XmlAttribute(AttributeName = "ITSVersion")]
    public string ITSVersion { get; set; }
}

public partial class RCMR_IN000030UV01
{
    // The added property
    [System.Xml.Serialization.XmlElementAttribute("distributionStatus", Namespace = "urn:bccdx.ca", IsNullable = false)]
    public BCCDXDistributionStatus distStatus { get; set; }
}

[System.Xml.Serialization.XmlTypeAttribute(TypeName = "BCCDX.DistributionStatus", Namespace = "urn:bccdx.ca")]
public partial class BCCDXDistributionStatus
{
    [System.Xml.Serialization.XmlElementAttribute("receivedTime", Namespace = "urn:bccdx.ca", IsNullable = false)]
    public TS receivedTime { get; set; }
}

public class TS
{
    [XmlAttribute("value")]
    public DateTime Value { get; set; }
}

目前您的服务正在返回如下所示的 XML:

<RCMR_IN000030UV01 ITSVersion="1.0"
    xmlns="urn:hl7-org:v3"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns:xsd="http://www.w3.org/2001/XMLSchema">
    <distributionStatus
        xmlns="urn:bccdx.ca">
        <receivedTime value="2017-02-23T00:00:00-05:00"/>
    </distributionStatus>
</RCMR_IN000030UV01>

但是,你想要这个:

<RCMR_IN000030UV01 ITSVersion="1.0"
    xmlns="urn:hl7-org:v3"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns:xsd="http://www.w3.org/2001/XMLSchema"
    <!---This should be added ---->
    xmlns:x="urn:bccdx.ca">
    <!---And distributionStatus should be prefixed with x: ---->
    <x:distributionStatus>
        <x:receivedTime value="2017-02-23T00:00:00-05:00"/>
    </x:distributionStatus>
</RCMR_IN000030UV01>

首先我会注意到这两个 XML 文件在语义上是相同的。在第一种情况下,命名空间"urn:bccdx.ca" 被声明为实际需要它的最低元素上的默认命名空间。在第二种情况下,它在文件的开头使用前缀进行定义。无论哪种方式,元素 &lt;distributionStatus&gt; 及其子元素都最终位于正确的命名空间中。

因此,您可以简单地接受 XML 原样正确。

如果出于某种原因,您必须让命名空间出现在 XML 的开头并带有 x: 前缀,您可以将 [XmlNamespaceDeclarations] 属性添加到您的 RCMR_IN000030UV01 以强制要在更高级别声明的命名空间:

public partial class RCMR_IN000030UV01
{
    [XmlNamespaceDeclarations]
    public XmlSerializerNamespaces xmlsn
    {
        get
        {
            var ns = new XmlSerializerNamespaces();
            ns.Add("x", "urn:bccdx.ca");
            return ns;
        }
        set
        {
            // Do nothing - fake property.
        }
    }
}

正如docs中解释的那样,这个属性

指定目标属性、参数、返回值或类成员包含与 XML 文档中使用的命名空间相关联的前缀。

现在您的服务应该根据需要返回带有根元素上的命名空间的 XML。

【讨论】:

  • 这是一个很好的答案,所有假设都已成立(我编辑了原始问题以包含 XmlSerializerFormat 声明)。我添加了建议的 [XmlNamespaceDeclarations] 属性并且它有效。我同意这两种情况下的 XML 都是正确的,但规范的一部分就是这样。
猜你喜欢
  • 1970-01-01
  • 2021-06-27
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2011-03-06
相关资源
最近更新 更多