为数组 属性 指定 xsi:type

Specify xsi:type for array property

我正在编写一个使用 XmlSerializer 的 WCF 服务。该服务是一个基于 WSDL 的实现和一个 XSD,它是从将使用该服务的组中提供给我的。基本上它是他们的系统和我的系统之间的数据适配器。

一个特定的 class 有一个 属性,它是项目中定义的另一种引用类型的数组。我需要为此 属性.

指定 xsi:type

我使用 svcutil 从 WSDL 和 XSD 生成代码,然后在几个地方 "fixed" 生成代码。这个 属性 是我必须修复的一个。

class 定义是(缩减到问题区域):

[GeneratedCode("svcutil", "4.6.1055.0")]
[Serializable]
[DebuggerStepThrough]
[DesignerCategory("code")]
[XmlType(Namespace = "urn:example.com:types")]
public class userData
{
    private ItemType[] itemTypeField;

    [XmlArray(Order = 0, Namespace = "urn:example.com:types")]
    [XmlArrayItem("item", IsNullable = false, Type = typeof(ItemType), Namespace = "urn:example.com:types")]
    public ItemType[] myprop
    {
        get { return itemTypeField; }
        set { itemTypeField = value; }
    }
}

调用服务方法时产生的XML是(userDatagetUserResponseclass的属性):

<s:Envelope xmlns:s="http://schemas.xmlsoap.org/soap/envelope/">
   <s:Body xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema">
      <getUserResponse xmlns="urn:example.com:types">
         <userData>
            <myprop>
               <item>
                  <key>some key</key>
                  <value>some value</value>
               </item>
               <item>
                  <key>some other key</key>
                  <value>some other value</value>
               </item>
            </myprop>
         </userData>
      </getUserResponse>
   </s:Body>
</s:Envelope>

我需要 <myprop> 元素看起来像这样:

<myprop xsi:type="ns1:MapType"
    xmlns:ns1="urn:example.com:types"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">

如何让 XmlSerializer 为数组 属性 myprop 生成 xsi:type

我发现这些类似的问题最接近我的问题,但它们不适用于数组:
xsi type and xsd tag
How can I force the use of an xsi:type attribute?

我需要使用自定义序列化程序来执行此操作吗?

如果我需要 post 任何服务合同,请告诉我。服务中的其他一切都工作正常,只是一点点。

我找不到使用任何内置机制(属性、配置等)来执行此操作的任何方法,所以我最终编写了一个自定义消息检查器来在 XML在响应中发出。这是代码:

自定义检查器 - 执行 XML 重写以添加所需的标记

using System.IO;
using System.ServiceModel;
using System.ServiceModel.Channels;
using System.ServiceModel.Dispatcher;
using System.Xml;

namespace MyProj
{
    public class CustomInspector : IDispatchMessageInspector
    {
        public object AfterReceiveRequest(ref Message request, IClientChannel channel, InstanceContext instanceContext)
        {
            return null;
        }

        public void BeforeSendReply(ref Message reply, object correlationState)
        {
            if (reply.Headers.Action == "urn:example:wsdl/Adapter/getUserResponse")
            {
                MessageBuffer copy = reply.CreateBufferedCopy(int.MaxValue);
                XmlDictionaryReader xdr = copy.CreateMessage().GetReaderAtBodyContents();
                XmlDocument doc = new XmlDocument();
                doc.Load(xdr);
                xdr.Close();

                doc.InnerXml = doc.InnerXml.Replace(@"<myprop>",
                    @"<myprop xsi:type=""ns1:MapType"" xmlns:ns1=""urn:example.com:types"" xmlns:xsi=""http://www.w3.org/2001/XMLSchema-instance"">");
                doc.InnerXml = doc.InnerXml.Replace(@"<item>", @"<item xmlns=""urn:example.com:types"">");

                MemoryStream ms = new MemoryStream();
                XmlWriter xw = XmlWriter.Create(ms);
                doc.Save(xw);
                xw.Flush();
                xw.Close();
                ms.Position = 0;
                XmlReader reader = XmlReader.Create(ms);


                Message newMsg = Message.CreateMessage(reply.Version, reply.Headers.Action, reader);
                reply = newMsg;
            }

        }
    }
}

自定义端点行为 - 将新消息检查器附加到服务端点:

using System.ServiceModel.Channels;
using System.ServiceModel.Description;
using System.ServiceModel.Dispatcher;

namespace MyProj
{
    public class CustomBehavior : IEndpointBehavior
    {
        public void Validate(ServiceEndpoint endpoint)
        {
        }

        public void AddBindingParameters(ServiceEndpoint endpoint, BindingParameterCollection bindingParameters)
        {
        }

        public void ApplyDispatchBehavior(ServiceEndpoint endpoint, EndpointDispatcher endpointDispatcher)
        {
            endpointDispatcher.DispatchRuntime.MessageInspectors.Add(new CustomInspector());
        }

        public void ApplyClientBehavior(ServiceEndpoint endpoint, ClientRuntime clientRuntime)
        {
        }
    }
}

自定义行为扩展元素 - 定义新行为

using System;
using System.ServiceModel.Configuration;

namespace KeypadIdmAdaptor
{
    public class CustomBehaviorExtensionElement : BehaviorExtensionElement
    {
        protected override object CreateBehavior()
        {
            return new CustomBehavior();
        }

        public override Type BehaviorType
        {
            get { return typeof(CustomBehavior); }
        }
    }
}

在 web.config 中连接它:

  <system.serviceModel>
    <extensions>
      <behaviorExtensions>
        <add name="CustomBehaviorExtension" type="MyProj.CustomBehaviorExtensionElement, MyProj, Version=1.0.0.0, Culture=neutral" />
      </behaviorExtensions>
    </extensions>
    <behaviors>
      <endpointBehaviors>
        <behavior>
          <CustomBehaviorExtension />
        </behavior>
      </endpointBehaviors>
      <serviceBehaviors>
        <behavior>
          <serviceMetadata httpGetEnabled="true" httpsGetEnabled="false" />
          <serviceDebug includeExceptionDetailInFaults="false" />
        </behavior>
      </serviceBehaviors>
    </behaviors>
    <services>
      <service name="MyProj.MyService">
        <endpoint binding="mexHttpBinding" address="mex" contract="IMetadataExchange"></endpoint>
        <endpoint name="MyServiceEndpoint" binding="basicHttpBinding" contract="MyProjServiceContract"></endpoint>
      </service>
    </services>
    <serviceHostingEnvironment aspNetCompatibilityEnabled="true" multipleSiteBindingsEnabled="true" />
  </system.serviceModel>

这里是响应消息中产生的XML:

<s:Envelope xmlns:s="http://schemas.xmlsoap.org/soap/envelope/">
   <s:Body>
      <getUserResponse xmlns="urn:example.com:types">
         <userData>
            <myProp xsi:type="ns1:MapType" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:ns1="urn:example.com:types">
               <item>
                  <key>some key</key>
                  <value>some value</value>
               </item>
               <item>
                  <key>some other key</key>
                  <value>some other value</value>
               </item>
            </myProp >
         </userData>
      </getUserResponse>
   </s:Body>
</s:Envelope>