我开发了一些复杂的WCF服务方法。我想使用流传输模式,因为我有多个参数,我已经定义了一个带有正文和标题的MessageContract。
[MessageContract]
public class ReportAudioMessage
{
[MessageHeader]
public int ReportId;
[MessageHeader]
public string FileName;
[MessageHeader]
public int FileLengthInBytes;
[MessageHeader]
public int LengthInSeconds;
[MessageBodyMember]
public Stream ReportAudio;
}
请注意,根据我在MSDN上阅读的指南,流是正文的唯一成员。
方法定义如下:
[OperationContract]
void SaveReportAudio(ReportAudioMessage reportToSave);
当我尝试调用方法(使用反射)时,出现错误:
反序列化操作请求消息正文时出错 'SaveReportAudio'。 OperationFormatter遇到无效的消息 身体。预计会找到名为'SaveReportAudio'的节点类型'Element' 和命名空间“http://tempuri.org/”。找到节点类型'Element' 名称“ReportAudioMessage”和名称空间“http://tempuri.org/”
SaveReportAudio是我正在调用的服务方法的名称。 ReportAudioMessage是已定义的MessageContract的名称。显然,我的肥皂消息正在被抬高,但我不知道如何...... :(
以下是服务的Web配置的服务模型节点:
<system.serviceModel>
<serviceHostingEnvironment multipleSiteBindingsEnabled="false" />
<bindings>
<netTcpBinding>
<binding
name="VRManagerTcpBinding"
closeTimeout="00:01:00"
openTimeout="00:01:00"
sendTimeout="00:01:00"
receiveTimeout="00:01:00"
transferMode="Streamed">
<reliableSession enabled="false"/>
<security mode="None" />
</binding>
</netTcpBinding>
</bindings>
<services>
<service name="Radia.VoiceRecognition.Services.VRManager" behaviorConfiguration="VRManagerTcpBehavior">
<host>
<baseAddresses>
<add baseAddress="net.tcp://localhost:8011/VRManager"/>
</baseAddresses>
</host>
<endpoint
address="VRManager.svc"
binding="netTcpBinding"
bindingConfiguration="VRManagerTcpBinding"
contract="Radia.VoiceRecognition.Services.IVRManager" />
<endpoint address="mex" binding="mexTcpBinding" contract="IMetadataExchange" />
</service>
</services>
<behaviors>
<serviceBehaviors>
<behavior name="VRManagerTcpBehavior">
<serviceMetadata httpGetEnabled="false" />
<serviceDebug includeExceptionDetailInFaults="true" />
</behavior>
</serviceBehaviors>
</behaviors>
</system.serviceModel>
这是客户端App.Config的服务模型节点:
<system.serviceModel>
<bindings>
<netTcpBinding>
<binding name="NetTcpBinding_IVRManager" closeTimeout="00:01:00"
openTimeout="00:01:00" receiveTimeout="00:10:00" sendTimeout="00:01:00"
transactionFlow="false" transferMode="Streamed" transactionProtocol="OleTransactions"
hostNameComparisonMode="StrongWildcard" listenBacklog="10" maxBufferPoolSize="524288"
maxBufferSize="65536" maxConnections="10" maxReceivedMessageSize="65536">
<readerQuotas maxDepth="32" maxStringContentLength="8192" maxArrayLength="16384"
maxBytesPerRead="4096" maxNameTableCharCount="16384" />
<reliableSession ordered="true" inactivityTimeout="00:10:00"
enabled="false" />
<security mode="None">
<transport clientCredentialType="Windows" protectionLevel="EncryptAndSign" />
<message clientCredentialType="Windows" />
</security>
</binding>
</netTcpBinding>
</bindings>
<client>
<endpoint address="net.tcp://xxxxxxxxxxx:8012/VRManager.svc/VRManager.svc"
binding="netTcpBinding" bindingConfiguration="NetTcpBinding_IVRManager"
contract="VRManager.IVRManager" name="NetTcpBinding_IVRManager" />
</client>
答案 0 :(得分:1)
虽然我对解决方案不满意,但我能够正常工作。我觉得这是一个黑客,部分是因为我完全不了解它。
在思考错误之后,我决定启用WCF跟踪并查看我能找到的内容。我看到了Xml消息并注意到该元素确实具有名称“ReportAudioMessage”。所以,我决定修改消息合约,并设置WrapperName:
[MessageContract(IsWrapped = true, WrapperName = "SaveReportAudio")]
public class ReportAudioMessage
{
[MessageBodyMember]
public Stream Session;
}
注意“WrapperName”属性。现在,这是我的WCF服务上的方法的名称。它现在有效。然而,令我感到沮丧的是,因为现在可以使用两种方法 - 保存方法和get方法:
[OperationContract]
void SaveReportAudio(ReportAudioMessage message);
[OperationContract]
ReportAudioMessage GetReportAudio(GetReportAudioRequestMessage request);
无论如何,它对我有用,所以我就是这么做的。我将不胜感激任何进一步的意见或建议。谢谢,