我正在通过Xamarin / MonoTouch(因此使用C#而不是Java)构建Android应用,并希望在该应用与服务器通信时显示ProgressBar。我一直在尝试使用异步来执行此操作,但是我对线程的工作方式没有很好的了解,并且在过去的几个小时中我一直遇到相同的问题-ProgressBar在我的方法调用之后而不是之前显示。我的代码在OnCreate()方法中,该方法也被重写为异步方法。这里是:
loginButton.Click += async (sender, e) =>
{
progbar.Visibility = ViewStates.Visible;
var userFetcher = new UserFetcher();
var json = await userFetcher.FetchUserDetailsAsync(/*parameters*/);
//the above method is async and returns a Task<JsonValue>
ParseUserDetails(json); //another method I created
progbar.Visibility = ViewStates.Invisible;
//do some stuff using the parsed data
};
我遇到的问题是FetchUserDetailsAsync似乎阻塞了我的线程。我一直在通过关闭服务器进行测试,以至于需要很长的响应时间(但是即使我测试了诸如Thread.Sleep(5000)之类的东西,我也遇到了同样的问题)。调用该方法后,它会同时运行
progbar.Visibility = ViewStates.Visible;
和progbar.Visibility = ViewStates.Invisible;
-我知道这是因为当我注释掉Invisible部分时,ProgressBar会在我的方法从服务器获得“响应”后出现。编译器还向我提供了诸如“跳过67帧!该应用程序可能在其主线程上做太多工作”的消息。就像我之前说的,我对线程没有真正的经验,所以很可能我只是天真地做错了什么。有谁能解决这个问题?
编辑:这是FetchUserDetailsAsync的源代码:
public async Task<JsonValue> FetchUserDetailsAsync(string url, string username, string password)
{
HttpWebRequest request = (HttpWebRequest)HttpWebRequest.Create(new Uri(url));
request.ContentType = "application/json";
request.Method = "GET";
string myBasicHeader = AuthenticationHelper.MakeHeader(username, password);
request.Headers.Add("Authorization", "Basic " + myBasicHeader);
try
{
using (HttpWebResponse response = (HttpWebResponse)request.GetResponse())
{
using (Stream stream = response.GetResponseStream())
{
JsonValue jsonDoc = await Task.Run(() => JsonObject.Load(stream));
return jsonDoc;
}
}
}
catch (WebException e)
{
Console.WriteLine(e.ToString());
if (e.Status == WebExceptionStatus.ProtocolError)
{
var response = e.Response as HttpWebResponse;
if (response != null)
{
Console.WriteLine("HTTP Status Code: " + (int)response.StatusCode);
}
else
{
Console.WriteLine("No http status code available");
}
}
return null;
}
}
最佳答案
看起来FetchUserDetailsAsync或ParseUserDetails阻止了UI线程。因为Task.Sleep(5000)正在同步运行。尝试等待Task.Delay(5000);在FetchUserDetailsAsync行中查看进度条是否显示。如果是这样,那么您可能需要进入FetchUserDetailsAsync实现以确保它是异步实现的