如何通过post将xml发送到wcf服务
本文关键字:wcf 服务 xml 何通过 post | 更新日期: 2023-09-27 17:59:57
我得到了一个将xml发布到wcf服务的代码。这是的完整代码
1) WCF服务接口
[OperationContract]
[WebInvoke(Method = "POST",
UriTemplate = "GetData",
RequestFormat = WebMessageFormat.Xml,
BodyStyle = WebMessageBodyStyle.Bare)]
string GetData(DataRequest parameter);
2) WCF服务实现
public string GetData(DataRequest parameter)
{
//Do stuff
return "your data here";
}
3) WCF服务中的数据合约(在本例中为DataRequest)
[DataContract(Namespace = "YourNamespaceHere")]
public class DataRequest
{
[DataMember]
public string ID{ get; set; }
[DataMember]
public string Data{ get; set; }
}
4) 发送数据的客户端必须正确构建数据!(本例中为C#控制台应用程序)
static void Main(string[] args)
{
ASCIIEncoding encoding = new ASCIIEncoding();
string SampleXml = "<DataRequest xmlns='"YourNamespaceHere'">" +
"<ID>" +
yourIDVariable +
"</ID>" +
"<Data>" +
yourDataVariable +
"</Data>" +
"</DataRequest>";
string postData = SampleXml.ToString();
byte[] data = encoding.GetBytes(postData);
string url = "http://localhost:62810/MyService.svc/GetData";
string strResult = string.Empty;
// declare httpwebrequet wrt url defined above
HttpWebRequest webrequest = (HttpWebRequest)WebRequest.Create(url);
// set method as post
webrequest.Method = "POST";
// set content type
webrequest.ContentType = "application/xml";
// set content length
webrequest.ContentLength = data.Length;
// get stream data out of webrequest object
Stream newStream = webrequest.GetRequestStream();
newStream.Write(data, 0, data.Length);
newStream.Close();
//Gets the response
WebResponse response = webrequest.GetResponse();
//Writes the Response
Stream responseStream = response.GetResponseStream();
StreamReader sr = new StreamReader(responseStream);
string s = sr.ReadToEnd();
return s;
}
我的问题是,如果GetData()函数需要两个或多个参数,那么我如何为GetData(()提供值
public string GetData(string xml1,string xml2)
{
//Do stuff return "your data here";
}
那么请指导我如何将两个xml数据传递给GetData()
函数?
首先,不能有一个带有WebMessageBodyStyle.Bare
的服务和一个使用两个参数的方法。您至少需要更改为WrappedRequest
:
[OperationContract]
[WebInvoke(Method = "POST",
UriTemplate = "GetData",
RequestFormat = WebMessageFormat.Xml,
BodyStyle = WebMessageBodyStyle.WrappedRequest)]
string GetData(DataRequest xml1, string xml2);
有了这些,让我们看看这个webHttp
绑定需要什么样的有线格式。
根元素需要命名为操作的名称GetData
。接下来是一系列子元素,每个参数一个。这些元素的默认名称等于变量的名称,因此在本例中为xml1
和xml2
。使用[MessageParameter(Name="Bar")]
属性装饰参数可以影响这些默认值。
应用上述规则将要求您的HTTPPost主体看起来类似于以下内容:
<GetData> <!-- Operation -->
<xml1> <!-- parameter one -->
</xml1>
<xml2> <!-- parameter two -->
</xml2>
</GetData>
不幸的是,事实并非如此。服务及其合约都有命名空间。在没有任何修饰的情况下,名称空间是丑陋的http://tempuri.org/
,我在IService
接口上用替换了它
[ServiceContract(Namespace="http://service.stackoverflow.com")]
在您的第一个示例中,DataRequest
类型有一个名称空间。在我的测试中,我用装饰了这个型号
[DataContract(Namespace="http://data.stackoverflow.com")]
现在,我们需要将这些名称空间(带有名称空间别名)添加到我们的基本xml负载中:
<GetData xmlns="http://service.stackoverflow.com"> <!-- Operation with default namespace -->
<xml1 xmlns:a="http://data.stackoverflow.com"> <!-- namespace on type -->
<a:ID>42</a:ID> <!-- members are in the type namespace, use the alias -->
<a:Data>FuBar</a:Data>
</xml1>
<xml2> <!-- parameter two, no explicit namespace so is in the default one -->
BlahBlah
</xml2>
</GetData>
在Service1.svc/GetData
向服务发布该有效载荷确实使用具有反序列化数据的参数来调用服务。
要回答您的问题,请准确地给出您在服务中更改代码的示例:
[ServiceContract(Namespace="http://service.stackoverflow.com")]
public interface IService1
{
[OperationContract]
[WebInvoke(Method = "POST",
UriTemplate = "GetData",
RequestFormat = WebMessageFormat.Xml,
BodyStyle = WebMessageBodyStyle.WrappedRequest)]
string GetData([MessageParameter(Name="Bar")] DataRequest xml1, string xml2);
}
在客户端:
var yourIDVariable = "foo";
var yourDataVariable = "bar";
string SampleXml = "<xml1 xmlns:a='"http://data.stackoverflow.com'">" +
"<a:ID>" +
yourIDVariable +
"</a:ID>" +
"<a:Data>" +
yourDataVariable +
"</a:Data>" +
"</xml1>";
string xml2 = "some data";
string wrapper = "<GetData xmlns='"http://service.stackoverflow.com'">{0}<xml2>{1}</xml2></GetData>";
string postData = String.Format(wrapper, SampleXml, xml2);
在处理WCF服务调用和诊断序列化问题时,在system.serviceModel
:的配置中启用WCF消息日志记录非常方便
<diagnostics>
<endToEndTracing activityTracing="true" messageFlowTracing="true" propagateActivity="true"/>
<messageLogging
logKnownPii="true"
logEntireMessage="true"
logMalformedMessages="true"
logMessagesAtServiceLevel="true"
logMessagesAtTransportLevel="true"
maxMessagesToLog="30000"
maxSizeOfMessageToLog="20000"/>
</diagnostics>
以及合适的听众:
<system.diagnostics>
<sources>
<source name="System.ServiceModel.MessageLogging">
<listeners>
<add name="messages"
type="System.Diagnostics.XmlWriterTraceListener"
initializeData="messages.svclog" />
</listeners>
</source>
</sources>
<trace autoflush="true"></trace>
</system.diagnostics>
它在服务跟踪查看器中为您提供了确切的有效载荷: