鉴于此WebApi服务:
[ActionName("KillPerson")]
[HttpPost]
public void KillPerson([FromBody] long id)
{
// Kill
}
这个HttpClient PostAsync调用:
var httpClient = new HttpClient { BaseAddress = new Uri(ClientConfiguration.ApiUrl) };
httpClient.DefaultRequestHeaders.Accept.Add(new MediaTypeWithQualityHeaderValue("application/json"));
var serializerSettings = new JsonSerializerSettings
{
PreserveReferencesHandling = PreserveReferencesHandling.Objects,
Formatting = Formatting.Indented,
ReferenceLoopHandling = ReferenceLoopHandling.Serialize
};
var serializedParameter = JsonConvert.SerializeObject(parameter, serializerSettings);
var httpContent = new StringContent(serializedParameter, Encoding.UTF8, "application/json");
var response = await httpClient.PostAsync(serviceUrl, httpContent).ConfigureAwait(false);
response.EnsureSuccessStatusCode();
我希望response.EnsureSuccessStatusCode();成功,但它会抛出404。 有趣的是,fiddler告诉med,webapi服务按预期返回204,当我调试它时,KillPerson运行没有问题。
更新: 我已确定只有在客户端代码位于PCL或Silverlight 5项目中时才会发生这种情况。如果我在Windows窗体应用程序中复制它,完全相同的代码将给出预期的204。如果我将Windows窗体应用程序指向PCL中包含的客户端代码,它会再次给我404。
UPDATE2: 这解决了这个问题(尽管让我感到困扰,我不应该这样做):
[ActionName("KillPerson")]
[HttpPost]
public HttpResponseMessage KillPerson([FromBody] long id)
{
return this.Request.CreateResponse(HttpStatusCode.OK);
}
重新推出404(小提琴手仍称204和非银光客户端运行正常)
[ActionName("KillPerson")]
[HttpPost]
public HttpResponseMessage KillPerson([FromBody] long id)
{
return this.Request.CreateResponse(HttpStatusCode.NoContent);
}
更新3(已解决): 终于搞清楚了。似乎您可以选择在Silverlight中使用浏览器或客户端HTTP处理。使用浏览器HTTP处理时,许多内容都不受支持 - 包括各种响应代码和标头。 在调用HttpClient之前添加这些行修复它:
WebRequest.RegisterPrefix("http://", WebRequestCreator.ClientHttp);
WebRequest.RegisterPrefix("https://", WebRequestCreator.ClientHttp);
答案 0 :(得分:4)
终于弄明白了。似乎您可以选择在Silverlight中使用浏览器或客户端HTTP处理。使用浏览器HTTP处理时,许多内容都不受支持 - 包括各种响应代码和标头。在调用HttpClient之前添加这些行修复它:
WebRequest.RegisterPrefix("http://", WebRequestCreator.ClientHttp);
WebRequest.RegisterPrefix("https://", WebRequestCreator.ClientHttp);
答案 1 :(得分:1)
将HttpResponseMessage返回类型添加到方法中是记录的方法,因此您找到的解决方案确实是最好的。
但是,如果您不想以这种方式更改所有的void方法,可以选择添加委托处理程序来动态更改响应代码 - 如下所示:
public class ResponseHandler : DelegatingHandler
{
protected override Task<HttpResponseMessage> SendAsync(HttpRequestMessage request, CancellationToken cancellationToken)
{
var response = base.SendAsync(request, cancellationToken);
if (request.Method == HttpMethod.Post)
{
response.Result.StatusCode = response.Result.IsSuccessStatusCode ? System.Net.HttpStatusCode.OK : response.Result.StatusCode;
}
return response;
}
}
注意:这会更改所有您的帖子方法的状态代码(成功时)。如果您只想为特定路线/方法完成,则必须添加一些代码(如果您需要不同的后处理方法,则需要区别对待)。