Android 非同步更新UI----handler+thread 中我們使用的與handler作為thread和UI線程的橋樑,android本身也給我們提供來一個非同步更新的方法AsyncTask, 其中AsyncTask比hander更輕量級一些。在Android1.5中提供了AsyncTask。它使建立需要與使用者介面互動的長時間啟動並執行任務變得更簡單。AsyncTask 的優勢體現在:•線程的開銷較大,如果每個任務都要建立一個線程,那麼應用程 序的效率要低很多; •線程無法管理,匿名線程建立並啟動後就不受程式的控制了,如果有很多個請求發送,那麼就會啟動非常多的線程,系統將不堪重負。 •另外,前面已經看到,UI還在新線程中更新必須要引入handler,這讓代碼看上去非常臃腫。當然AsyncTask也不是沒有缺點的,這個在下一章中詳細介紹。 AsyncTask抽象出後台線程啟動並執行五個狀態,分別是:1、準備運行,2、正在後台運行,3、進度更新,4、完成背景工作,5、取消任務,對於這五個階段,AsyncTask提供了五個回呼函數: 1、準備運行:onPreExecute(),該回呼函數在任務被執行之後立即由UI線程調用。這個步驟通常用來建立任務,在使用者介面(UI)上顯示進度條。 2、正在後台運行:doInBackground(Params...),該回呼函數由後台線程在onPreExecute()方法執行結束後立即調用。通常在這裡執行耗時的後台計算。計算的結果必須由該函數返回,並被傳遞到onPostExecute()中。在該函數內也可以使用publishProgress(Progress...)來發布一個或多個進度單位(unitsof progress)。這些值將會在onProgressUpdate(Progress...)中被發布到UI線程。 3. 進度更新:onProgressUpdate(Progress...),該函數由UI線程在publishProgress(Progress...)方法調用完後被調用。一般用於動態地顯示一個進度條。 4. 完成背景工作:onPostExecute(Result),當後台計算結束後調用。後台計算的結果會被作為參數傳遞給這一函數。 5、取消任務:onCancelled (),在調用AsyncTask的cancel()方法時調用 AsyncTask的建構函式有三個模板參數: 1.Params,傳遞給背景工作的參數類型。 2.Progress,後台計算執行過程中,進步單位(progress units)的類型。(就是背景程式已經執行了百分之幾了。) 3.Result, 後台執行返回的結果的類型。 AsyncTask並不總是需要使用上面的全部3種類型。標識不使用的類型很簡單,只需要使用Void類型即可。
package com.example.thread; import org.apache.http.HttpResponse; import org.apache.http.client.HttpClient; import org.apache.http.client.methods.HttpGet; import org.apache.http.impl.client.DefaultHttpClient; import com.example.test.R; import android.app.Activity; import android.graphics.Bitmap; import android.graphics.BitmapFactory; import android.os.AsyncTask; import android.os.Bundle; import android.view.View; import android.view.View.OnClickListener; import android.widget.Button; import android.widget.ImageView; import android.widget.ProgressBar; import android.widget.Toast; public class AsyncTaskActivity extends Activity{ private ImageView mImageView; private Button mButton; private ProgressBar mProgressBar; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main12); mImageView = (ImageView)findViewById(R.id.imageView); mButton = (Button)findViewById(R.id.button); mProgressBar = (ProgressBar) findViewById(R.id.progressBar); mButton.setOnClickListener(new OnClickListener() { @Override public void onClick(View v) { new MyAsyncTask().execute("http://csdnimg.cn/www/images/csdnindex_logo.gif"); } }); } class MyAsyncTask extends AsyncTask<String, Integer, Bitmap>{ @Override protected Bitmap doInBackground(String... params) { publishProgress(0);//將會調用onProgressUpdate(Integer... progress)方法 HttpClient hc = new DefaultHttpClient(); publishProgress(30); HttpGet hg = new HttpGet(params[0]); final Bitmap bm; try { HttpResponse hr = hc.execute(hg); bm = BitmapFactory.decodeStream(hr.getEntity().getContent()); } catch (Exception e) { return null; } publishProgress(100); return bm; } @Override protected void onProgressUpdate(Integer... values) { mProgressBar.setProgress(values[0]); } @Override protected void onPostExecute(Bitmap result) { if (result != null) { Toast.makeText(AsyncTaskActivity.this, "成功擷取圖片", Toast.LENGTH_LONG).show(); mImageView.setImageBitmap(result); }else { Toast.makeText(AsyncTaskActivity.this, "擷取圖片失敗", Toast.LENGTH_LONG).show(); } } @Override protected void onPreExecute() { mImageView.setImageBitmap(null); mProgressBar.setProgress(0); super.onPreExecute(); } @Override protected void onCancelled() { super.onCancelled(); mProgressBar.setProgress(0); } } }