06月12, 2014

Android中AyncTask数据实时刷新同步问题

当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<Void, Integer, String> {
    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<Void, Integer, String> {
        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;
            }
        }
......
}

本文链接:http://blogs.360.cn/post/android中aynctask数据实时刷新同步问题.html

-- EOF --

Comments