我有一个 AJAX 调用,我想针对 WCF GET 服务运行该调用。基本上,对服务的调用(通过 jquery)如下所示:
$.get(serviceEndpoint, {query : "some search text", statusTypes: [1, 2]}, function (result) { /* do something*/ }, 'text');
当这个调用运行时,我看到 firebug 中的 GET 正确通过,并且我确实到达了端点。然而,参数statusTypes
始终为空。
来自 jquery 的 GET 本身看起来像是经过编码的,但是当我不对括号进行编码时,调用根本不会进入端点:
以及 WCF 服务本身:
【运营合同】
[WebInvoke(Method= "GET", BodyStyle = WebMessageBodyStyle.WrappedRequest,
响应格式 =
WebMessageFormat.Json)]
民众
结果视图模型
GetTags(字符串查询, int[]
状态类型)
是否可以通过 GET 将数组传递给 WCF 服务?
排列并不多,因此我可以“每个数组”编写一个单独的端点,但我宁愿将其保留在一个端点中。
这是可能的,但对于开箱即用的 WCF 来说是不可能的。随着“jQuery 支持”WCF 代码复合页,您可以在无类型变量中接收 jQuery 发送的所有数据(包括数组、嵌套对象等),无论是在正文(对于 POST 请求)还是在查询字符串(对于 GET)上。 jQuery 数组变量(其名称包含“[”和“]”)和操作参数之间的映射无法在 WCF 4.0 中完成(至少在不编写消息格式化程序的情况下无法完成)。
然而,在新的 WCF Web API(也可在 codeplex 站点上找到)上,这应该更简单。
Update:这是适合您的场景的格式化程序示例:
public class StackOverflow_6445171
{
[ServiceContract]
public class Service
{
[WebGet(ResponseFormat = WebMessageFormat.Json)]
public string GetLabelPacketTags(string query, int[] statusTypes)
{
StringBuilder sb = new StringBuilder();
sb.Append("Query=" + query);
sb.Append(", statusTypes=");
if (statusTypes == null)
{
sb.Append("null");
}
else
{
sb.Append("[");
for (int i = 0; i < statusTypes.Length; i++)
{
if (i > 0) sb.Append(",");
sb.Append(statusTypes[i]);
}
sb.Append("]");
}
return sb.ToString();
}
}
class MyWebHttpBehavior : WebHttpBehavior
{
protected override IDispatchMessageFormatter GetRequestDispatchFormatter(OperationDescription operationDescription, ServiceEndpoint endpoint)
{
return new MyArrayAwareFormatter(operationDescription, this.GetQueryStringConverter(operationDescription));
}
class MyArrayAwareFormatter : IDispatchMessageFormatter
{
OperationDescription operation;
QueryStringConverter queryStringConverter;
public MyArrayAwareFormatter(OperationDescription operation, QueryStringConverter queryStringConverter)
{
this.operation = operation;
this.queryStringConverter = queryStringConverter;
}
public void DeserializeRequest(Message message, object[] parameters)
{
if (message.Properties.ContainsKey("UriMatched") && (bool)message.Properties["UriMatched"])
{
UriTemplateMatch match = message.Properties["UriTemplateMatchResults"] as UriTemplateMatch;
NameValueCollection queryValues = match.QueryParameters;
foreach (MessagePartDescription parameterDescr in this.operation.Messages[0].Body.Parts)
{
string parameterName = parameterDescr.Name;
int index = parameterDescr.Index;
if (parameterDescr.Type.IsArray)
{
Type elementType = parameterDescr.Type.GetElementType();
string[] values = queryValues.GetValues(parameterName + "[]");
Array array = Array.CreateInstance(elementType, values.Length);
for (int i = 0; i < values.Length; i++)
{
array.SetValue(this.queryStringConverter.ConvertStringToValue(values[i], elementType), i);
}
parameters[index] = array;
}
else
{
parameters[index] = this.queryStringConverter.ConvertStringToValue(queryValues.GetValues(parameterName)[0], parameterDescr.Type);
}
}
}
}
public Message SerializeReply(MessageVersion messageVersion, object[] parameters, object result)
{
throw new NotSupportedException("This is a request-only formatter");
}
}
}
public static void Test()
{
string baseAddress = "http://" + Environment.MachineName + ":8000/Service";
ServiceHost host = new ServiceHost(typeof(Service), new Uri(baseAddress));
host.AddServiceEndpoint(typeof(Service), new WebHttpBinding(), "").Behaviors.Add(new MyWebHttpBehavior());
host.Open();
Console.WriteLine("Host opened");
WebClient c = new WebClient();
Console.WriteLine(c.DownloadString(baseAddress + "/GetLabelPacketTags?query=some+text&statusTypes[]=1&statusTypes[]=2"));
Console.WriteLine(c.DownloadString(baseAddress + "/GetLabelPacketTags?query=some+text&statusTypes%5B%5D=1&statusTypes%5B%5D=2"));
Console.Write("Press ENTER to close the host");
Console.ReadLine();
host.Close();
}
}
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)