AsyncTask应用解析

当第一次启动一个Android程序时,Android会自动创建一个称为“main”主线程的线程。这个主线程(也称为UI线程)很重要,因为它负责把事件分派到相应的控件,其中就包括屏幕绘图事件,它同样是用户与Andriod控件交互的线程。比如,当你在屏幕上按下一个按钮后,UI线程会把这个事件分发给刚按得那个按钮,紧接着按钮设置它自身为被按下状态并向事件队列发送一个无效(invalidate)请求。UI线程会把这个请求移出事件队列并通知按钮在屏幕上重新绘制自身。

Android单线程模型会在没有考虑到它的影响的情况下引起Android应用程序性能低下,因为所有的任务都在同一个线程中执行,如果执行一些耗时的操作,如访问网络或查询数据库,会阻塞整个用户界面。当在执行一些耗时的操作的时候,不能及时地分发事件,包括用户界面重绘事件。从用户的角度来看,应用程序看上去像挂掉了。更糟糕的是,如果阻塞应用程序的时间过长(现在大概是5秒钟)Android会向用户提示一些信息,即打开一个“应用程序没有相应(application not responding)”的对话框。在单线程模型中始终要记住两条法则:

不要阻塞UI线程

确保只在UI线程中访问Android UI工具包

Android之所以有Handler和AsyncTask,都是为了不阻塞主线程(UI线程),且UI的更新只能在主线程中完成,因此异步处理是不可避免的。

Android 1.5提供了一个工具类:AsyncTask,它使创建需要与用户界面交互的长时间运行的任务变得更简单。不需要借助线程和Handler即可实现。

AsyncTask 的优势体现在:

•线程的开销较大,如果每个任务都要创建一个线程,那么应用程 序的效率要低很多;
•线程无法管理,匿名线程创建并启动后就不受程序的控制了,如果有很多个请求发送,那么就会启动非常多的线程,系统将不堪重负。
•另外,前面已经看到,在新线程中更新UI还必须要引入handler,这让代码看上去非常臃肿。

AsyncTask定义了三种泛型类型 Params,Progress和Result。
    •Params 启动任务执行的输入参数,比如HTTP请求的URL。
    •Progress 后台任务执行的百分比。
    •Result 后台执行任务最终返回的结果,比如String。

AsyncTask的执行分为四个步骤,每一步都对应一个回调方法,开发者需要实现一个或几个方法。在任务的执行过程中,这些方法被自动调用。

子类化AsyncTask

实现AsyncTask中定义的下面一个或几个方法

onPreExecute(), 该方法将在执行实际的后台操作前被UI thread调用。可以在该方法中做一些准备工作,如在界面上显示一个进度条。
doInBackground(Params...), 将在onPreExecute 方法执行后马上执行,该方法运行在后台线程中。这里将主要负责执行那些很耗时的后台计算工作。可以调用 publishProgress方法来更新实时的任务进度。该方法是抽象方法,子类必须实现。
onProgressUpdate(Progress...),在publishProgress方法被调用后,UI thread将调用这个方法从而在界面上展示任务的进展情况,例如通过一个进度条进行展示。
onPostExecute(Result), 在doInBackground 执行完成后,onPostExecute 方法将被UI thread调用,后台的计算结果将通过该方法传递到UI thread.
使用AsyncTask类,以下是几条必须遵守的准则:
    1) Task的实例必须在UI thread中创建
    2) execute方法必须在UI thread中调用
    3) 不要手动的调用onPreExecute(), onPostExecute(Result),doInBackground(Params...), onProgressUpdate(Progress...)这几个方法
    4) 该task只能被执行一次,否则多次调用时将会出现异常

从网上获取一个网页,在一个TextView中将其源代码显示出来

  1. package test.list;  
  2. import java.io.ByteArrayOutputStream;  
  3. import java.io.InputStream;  
  4. import java.util.ArrayList;  
  5.  
  6. import org.apache.http.HttpEntity;  
  7. import org.apache.http.HttpResponse;  
  8. import org.apache.http.client.HttpClient;  
  9. import org.apache.http.client.methods.HttpGet;  
  10. import org.apache.http.impl.client.DefaultHttpClient;  
  11.  
  12. import android.app.Activity;  
  13. import android.app.ProgressDialog;  
  14. import android.content.Context;  
  15. import android.content.DialogInterface;  
  16. import android.os.AsyncTask;  
  17. import android.os.Bundle;  
  18. import android.os.Handler;  
  19. import android.os.Message;  
  20. import android.view.View;  
  21. import android.widget.Button;  
  22. import android.widget.EditText;  
  23. import android.widget.TextView;  
  24.  
  25. public class NetworkActivity extends Activity{  
  26.  private TextView message;  
  27.     private Button open;  
  28.     private EditText url;  
  29.  
  30.     @Override  
  31.     public void onCreate(Bundle savedInstanceState) {  
  32.        super.onCreate(savedInstanceState);  
  33.        setContentView(R.layout.network);  
  34.        message= (TextView) findViewById(R.id.message);  
  35.        url= (EditText) findViewById(R.id.url);  
  36.        open= (Button) findViewById(R.id.open);  
  37.        open.setOnClickListener(new View.OnClickListener() {  
  38.            public void onClick(View arg0) {  
  39.               connect();  
  40.            }  
  41.        });  
  42.  
  43.     }  
  44.  
  45.     private void connect() {  
  46.      PageTask task = new PageTask(this);  
  47.         task.execute(url.getText().toString());  
  48.     }  
  49.  
  50.  
  51.     class PageTask extends AsyncTask<String, Integer, String> {  
  52.         // 可变长的输入参数,与AsyncTask.exucute()对应  
  53.      ProgressDialog pdialog;  
  54.         public PageTask(Context context){  
  55.          pdialog = new ProgressDialog(context, 0);     
  56.          pdialog.setButton("cancel", new DialogInterface.OnClickListener() {  
  57.           public void onClick(DialogInterface dialog, int i) {  
  58.            dialog.cancel();  
  59.           }  
  60.          });  
  61.          pdialog.setOnCancelListener(new DialogInterface.OnCancelListener() {  
  62.           public void onCancel(DialogInterface dialog) {  
  63.            finish();  
  64.           }  
  65.          });  
  66.          pdialog.setCancelable(true);  
  67.          pdialog.setMax(100);  
  68.          pdialog.setProgressStyle(ProgressDialog.STYLE_HORIZONTAL);  
  69.          pdialog.show();  
  70.  
  71.  
  72.         }  
  73.         @Override  
  74.         protected String doInBackground(String... params) {  
  75.  
  76.             try{  
  77.  
  78.                HttpClient client = new DefaultHttpClient();  
  79.                // params[0]代表连接的url  
  80.                HttpGet get = new HttpGet(params[0]);  
  81.                HttpResponse response = client.execute(get);  
  82.                HttpEntity entity = response.getEntity();  
  83.                long length = entity.getContentLength();  
  84.                InputStream is = entity.getContent();  
  85.                String s = null;  
  86.                if(is != null) {  
  87.                    ByteArrayOutputStream baos = new ByteArrayOutputStream();  
  88.  
  89.                    byte[] buf = new byte[128];  
  90.  
  91.                    int ch = -1;  
  92.  
  93.                    int count = 0;  
  94.  
  95.                    while((ch = is.read(buf)) != -1) {  
  96.  
  97.                       baos.write(buf, 0, ch);  
  98.  
  99.                       count += ch;  
  100.  
  101.                       if(length > 0) {  
  102.                           // 如果知道响应的长度,调用publishProgress()更新进度  
  103.                           publishProgress((int) ((count / (float) length) * 100));  
  104.                       }  
  105.  
  106.                       // 让线程休眠100ms  
  107.                       Thread.sleep(100);  
  108.                    }  
  109.                    s = new String(baos.toByteArray());              }  
  110.                // 返回结果  
  111.                return s;  
  112.             } catch(Exception e) {  
  113.                e.printStackTrace();  
  114.  
  115.             }  
  116.  
  117.             return null;  
  118.  
  119.         }  
  120.  
  121.         @Override  
  122.         protected void onCancelled() {  
  123.             super.onCancelled();  
  124.         }  
  125.  
  126.         @Override  
  127.         protected void onPostExecute(String result) {  
  128.             // 返回HTML页面的内容  
  129.             message.setText(result);  
  130.             pdialog.dismiss();   
  131.         }  
  132.  
  133.         @Override  
  134.         protected void onPreExecute() {  
  135.             // 任务启动,可以在这里显示一个对话框,这里简单处理  
  136.             message.setText(R.string.task_started);  
  137.         }  
  138.  
  139.         @Override  
  140.         protected void onProgressUpdate(Integer... values) {  
  141.             // 更新进度  
  142.               System.out.println(""+values[0]);  
  143.               message.setText(""+values[0]);  
  144.               pdialog.setProgress(values[0]);  
  145.         }  
  146.  
  147.      }  
  148.  
  149. }  
posted @ 2011-07-29 23:41  Colin Xie  阅读(362)  评论(0编辑  收藏  举报