当Android中使用AyncTask来执行耗时操作——在最常用的方法doInBackground(Params…)中执行时,如果涉及到执行结果要影响到UI线程,比如在界面中显示后台执行的结果时,我们需要传递数据到UI线程处理,这就是传递数据到onPostExecute(Result)处理。但后台执行循环操作或边执行边传递到UI线程处理,就涉及到publishProgress(progress)传递后台执行的结果到UI线程实时刷新数据,且需要在onProgressUpdate(Progress…)中处理后台执行结果反应到UI的操作,而不是在onPostExecute(Result)处理。值得注意的时,在publishProgress(progress)中参数progress传递的数据最好为onProgressUpdate(Progress…)方法中要显示或利用的全部数据一起传递过去,这是为了别勉在界面显示或获取的数据不同步。还有个问题就是,在线程中避免数据不同步,最好不要采用线程中有线程。
比如,我在写一个测试工具时,需要处理批量的URL是否可访问,访问是否成功。这个批量的访问处理就是在线程中执行的(网络请求需在线程中执行,且批量的数据在线程中执行不会造成UI线程长时间无响应而ANR(理论上一般UI线程等待5s后没响应就会ANR))。我需要每执行完一个URL的访问就把结果传递到UI中处理实时刷新显示,利用publishProgress(progress)和onProgressUpdate(Progress…)方法很容易解决,但有个问题需要注意:就是UI中显示的所有实时数据是否就是当前URL访问的结果(因为传递的数据和显示的数据不是同步传递过去的)。
如以下两个列子的对比:(列二就很有可能造成显示在界面的数据不同步,造成的原因是:线程还在执行得出的结果reInteger传递到onProgressUpdate(Integer… values)中与在该方法中直接获取的数据HTTP + statusCode + str_http不一致)
例一:
class ExcuteHttp extends AsyncTask{
String reString = "访问结束";
Integer reInteger;
String str_http;
String h_tv = "";
String HTTP = "HTTP码:";
int statusCode;
protected String doInBackground(Void... params){
........
for (int i = 0; i < strings.length && !stopflaghttp; i++) {
.......
if (statusCode == HttpStatus.SC_OK) {
// reInteger = 0;
h_tv += HTTP + statusCode+"," + str_http + "——能成功访问"
+ "\n\n";
}
......
publishProgress(h_tv);
}
.......
return reString;
}
protected void onProgressUpdate(String... values){
super.onProgressUpdate(values);
tv_hTextView.setText(h_tv + "\n");
}
........
}
例二:
class ExcuteHttp extends AsyncTask{
String reString = "访问结束";
Integer reInteger;
String str_http;
String h_tv = "";
String HTTP = "HTTP码:";
int statusCode;
protected String doInBackground(Void... params){
........
for (int i = 0; i < strings.length; i++) {
........
if (statusCode == HttpStatus.SC_OK) {
reInteger = 0;
}
........
publishProgress(reInteger);
}// 线程中多次循环
...
return reString;
}
protected void onProgressUpdate(Integer... values){
super.onProgressUpdate(values);
switch (values[0]) {
case 0:
h_tv += HTTP + statusCode + str_http + "——能成功访问" + "\n\n";
tv_hTextView.setText(h_tv);
break;
}
}
......
}