AsyncTask
看上去修改後的connect()方法已經可用了,但是這種匿名線程的方式是存在缺陷的:第一,線程的開銷較大,如果每個任務都要建立一個線程,那麼應用
程式的效率要低很多;第二,線程無法管理,匿名線程建立並啟動後就不受程式的控制了,如果有很多個請求發送,那麼就會啟動非常多的線程,系統將不堪重負。 另外,前面已經看到,在新線程中更新UI還必須要引入handler,這讓代碼看上去非常臃腫。
為瞭解決這一問題,OPhone在1.5版本引入了AsyncTask。AsyncTask的特點是任務在主線程之外運行,而回調方法是在主線程中執行, 這就有效地避免了使用Handler帶來的麻煩。閱讀AsyncTask的源碼可知,AsyncTask是使用java.util.concurrent 架構來管理線程以及任務的執行的,concurrent架構是一個非常成熟,高效的架構,經過了嚴格的測試。這說明AsyncTask的設計很好的解決了
匿名線程存在的問題。
AsyncTask是抽象類別,子類必須實現抽象方法doInBackground(Params... p) ,在此方法中實現任務的執行工作,比如串連網路擷取資料等。通常還應該實現onPostExecute(Result r)方法,因為應用程式關心的結果在此方法中返回。需要注意的是AsyncTask一定要在主線程中建立執行個體。AsyncTask定義了三種泛型型別
Params,Progress和Result。
* Params 啟動任務執行的輸入參數,比如HTTP請求的URL。
* Progress 背景工作執行的百分比。
* Result 後台執行任務最終返回的結果,比如String。
AsyncTask 的執行分為四個步驟,與前面定義的TaskListener類似。每一步都對應一個回調方法,需要注意的是這些方法不應該由應用程式調用,開發人員需要做的就是實現這些方法。在任務的執行過程中,這些方法被自動調用。
* onPreExecute() 當任務執行之前開始調用此方法,可以在這裡顯示進度對話方塊。
* doInBackground(Params...) 此方法在後台線程執行,完成任務的主要工作,通常需要較長的時間。在執行過程中可以調用publicProgress(Progress...)來更新任務的進度。
* onProgressUpdate(Progress...) 此方法在主線程執行,用於顯示任務執行的進度。
* onPostExecute(Result) 此方法在主線程執行,任務執行的結果作為此方法的參數返回。
PageTask擴充了AsyncTask,在 doInBackground()方法中讀取網頁內容。PageTask的原始碼如下所示:
// 設定三種型別參數分別為String,Integer,String class PageTask extends AsyncTask<String, Integer, String> { // 可變長的輸入參數,與AsyncTask.exucute()對應 @Override protected String doInBackground(String... params) { try { HttpClient client = new DefaultHttpClient(); // params[0] 代表串連的url HttpGet get = new HttpGet(params[0]); HttpResponse response = client.execute(get); HttpEntity entity = response.getEntity(); long length = entity.getContentLength(); InputStream is = entity.getContent(); String s = null; if (is != null) { ByteArrayOutputStream baos = new ByteArrayOutputStream(); byte[] buf = new byte[128]; int ch = -1; int count = 0; while ((ch = is.read(buf)) != -1) { baos.write(buf, 0, ch); count += ch; if (length > 0) { // 如果知道響應的長度,調用publishProgress()更新進度 publishProgress((int) ((count / (float) length) * 100)); } // 為了在模擬器中清楚地看到進度,讓線程休眠100ms Thread.sleep(100); } s = new String(baos.toByteArray()); } // 返回結果 return s; } catch (Exception e) { e.printStackTrace(); } return null; } @Override protected void onCancelled() { super.onCancelled(); } @Override protected void onPostExecute(String result) { // 返回HTML頁面的內容 message.setText(result); } @Override protected void onPreExecute() { // 任務啟動,可以在這裡顯示一個對話方塊,這裡簡單處理 message.setText(R.string.task_started); } @Override protected void onProgressUpdate(Integer... values) { // 更新進度 message.setText(values[0]); } }
執行PageTask非常簡單,只需要調用如下代碼。重新運行NetworkActivity,不但可以抓取網頁的內容,還可以即時更新讀取的進度。讀者嘗試讀取一個較大的網頁,看看百分比的更新情況。
PageTask task = new PageTask(); task.execute(url.getText().toString());
源碼下載