await httpClient.SendAsync(httpContent)
没有响应,尽管我发现代码/url 中没有错误,但它仍然挂起。请建议/帮助。
我的代码如下:
public async Task<string> Get_API_Result_String(string url, List<KeyValuePair<string, string>> parameters)
{
string res = "";
try
{
IsolatedStorageSettings settings = IsolatedStorageSettings.ApplicationSettings;
//Prepare url
Uri mainurl = new Uri(settings[FSAPARAM.UserSettingsParam.SERVERNAME].ToString());
Uri requesturl = new Uri(mainurl, url);
var httpClient = new HttpClient();
var httpContent = new HttpRequestMessage(HttpMethod.Post, requesturl);
// httpContent.Headers.ExpectContinue = false;
httpContent.Content = new FormUrlEncodedContent(parameters);
HttpResponseMessage response = await httpClient.SendAsync(httpContent);
var result = await response.Content.ReadAsStringAsync();
res = result.ToString();
response.Dispose();
httpClient.Dispose();
httpContent.Dispose();
}
catch (Exception ex)
{
Logger l = new Logger();
l.LogInfo("Get_API_Result_String: "+ url + ex.Message.ToString());
ex = null;
l = null;
}
return res;
}
在另一个类中调用如下:
NetUtil u = new NetUtil();
string result = await u.Get_API_Result_String(Register_API, values);
u = null;
我预测,在调用堆栈的更上方,您将在返回的
Wait
上调用 Result
或 Task
。 这会导致死锁,我在我的博客上对此进行了详细解释。
总而言之,
await
将捕获上下文并使用它来恢复async
方法;在 UI 应用程序中,这是一个 UI 线程。但是,如果 UI 线程被阻塞(在调用 Wait
或 Result
中),则该线程无法恢复 async
方法。
这对我有用:
httpClient.SendAsync(httpContent).ConfigureAwait(false);
我刚刚删除了
await
并按如下方式使用,它起作用了:
var result = httpClient.SendAsync(httpContent).Result;
但这不是一个好的做法。
作为
Nikola提到,我们不应该混合同步和异步调用。
我将调用方法更改为异步,问题得到解决。
这对我来说没问题
var response = httpClient.SendAsync(request);
var responseResult = response.Result;
if (responseResult.IsSuccessStatusCode)
{
var result = responseResult.Content.ReadAsStringAsync().Result;
return result;
}
刚刚在返回压缩的 gzip 数据时遇到此错误。这是一种非常罕见的情况,因为 99% 的情况下,使用不同的输入数据一切都很好。不得不切换到 HttpWebRequest。
但是还有另一种不那么麻烦的方法——更好地以同步方式调用可等待方法,这样不会阻塞 UI 线程。
// UI level method.
public void Button1_Click(...)
{
var u = new NetUtil();
var result = Task.Run(async () => await u.Get_API_Result_String(Register_API, values)).GetAwaiter().GetResult();
}