Android 使用AsyncTask 後監聽非同步載入完畢的動作

weixin_34292959發表於2011-03-07

   AsyncTask 的使用方法網上有很多例子,使用起來也非常的方便。這裡就不詳細說具體的使用方法了,同學可以Google 一下,很多。

場景模擬

      當我們在載入一個列表的時候,比如GridView ,這時候我們考慮到不阻塞UI的做法,一般會使用執行緒Thread 、Timer 或者使用AsyncTask ,而這些操作都是在在後臺另外開一個執行緒給我們找資料,具體得到的資料需要使用Handler 去更新UI,AsyncTask 也是一樣使用到的Handler 只是它將Handler 封裝在了onPostExecute 執行操作中。而這一操作可能會產生一個問題,比如你有一個列表更新資料庫使用到的是AsyncTask 非同步操作的方式更新UI,而你的需求是當我一進來這個列表就統計這個列表的資料的數量或者讓某一行資料的狀態為選中狀態。傳統做法是直接new 一個AsyncTask 類讓它execute(); 之後會再操作UI。想法是對的,但有一個問題我們要注意到,因為它是非同步載入資料的方式,而你的資料量比較多或許查詢資料需要一定的時間的時候,這時使用AsyncTask 執行非同步載入後更新UI再操作UI物件,可能會報空指標。

      這個問題的產生是,我們都知道程式碼的執行是自上而下執行,當你使用非同步載入資料的時候,程式碼讓你去執行非同步操作就不管了(多執行緒),而繼續會往下執行程式碼,你下面的程式碼就是操作列表裡面的UI,這時可想而知,非同步載入資料還沒有結束還沒有對你的UI進行更新,這些你的列表應該是空的,而操作一個空的列表就會報空指標。

分析問題

 

   使用過AsyncTask 的同學都知道一個非同步載入資料最少要重寫以下這兩個方法:

  • doInBackground   後臺執行,比較耗時的操作都可以放在這裡。注意這裡不能直接操作UI。
  • onPostExecute   相當於Handler 處理UI的方式,在這裡面可以使用在doInBackground 得到的結果處理操作UI。

     有必要的話你還得重寫以下這三個方法,但不是必須的:

  • onProgressUpdate   可以使用進度條增加使用者體驗度。
  • onPreExecute           這裡是終端使用者呼叫Excute時的介面
  • onCancelled             使用者呼叫取消時,要做的操作

      根據上面的思路,可以看出,最終資料載入並顯示出來這一系列的操作都在onPostExecute  這個方法裡面,那麼如何監聽所有UI都己經在onPostExecute   裡面處理完成了,再去執行我們自己要操作呢?

 

解決問題

 

       這裡給出我自己解決這一問題的思路,有更好想法的朋友歡迎跟貼共同探討。

 

       首先建立一個介面

 

private interface isLoadDataListener {
        
public void loadComplete();
    }

     宣告這一介面變數

 

private isLoadDataListener loadLisneter;

 

  給介面賦值,得到介面物件

 

public void setLoadDataComplete(isLoadDataListener dataComplete) {
        
this.loadLisneter = dataComplete;
    }

 

之後在AsyncTask 的onPostExecute處理UI完成後呼叫該介面,下面給出一個我以前專案使用到的AsyncTask 類:

 

    class loadGridAsyncTask extends AsyncTask<Integer, Integer, AppsAdapter> {

        
private int poindex;

        
public loadGridAsyncTask(int positionindex) {
            
this.poindex = positionindex;
        }

        @Override
        
protected AppsAdapter doInBackground(Integer... params) {
            
// TODO Auto-generated method stub
            
// mAppsModel.clear();
            Cursor temp = dbHelper.queryPageById(poindex);
            loadPage(mApps, temp);
            temp.close();
            
return new AppsAdapter(STB.this, mAppsModel);
        }

        @Override
        
protected void onPostExecute(AppsAdapter result) {

            gridViewExt itemGrid 
= (gridViewExt) viewFlipper
                    .getChildAt(poindex);
            itemGrid.setColumnCount(pageColumnCount);
            itemGrid.setAdapter(result);
            
if (loadLisneter != null) {
                loadLisneter.loadComplete();
            }
             
        }

    }

 

       通過上面的程式碼,我們就得到一個資料載入完成後返回的介面,接下來的問題就是我們利用這個介面來處理我們的UI了,比如讓某一UI選中,得到這個列表的UI數量等,看下面的程式碼:

 

new loadGridAsyncTask(1).execute();
                                    setLoadDataComplete(
new isLoadDataListener() {

                                        @Override
                                        
public void loadComplete() {
                                            
// TODO Auto-generated method stub
                                             
//這裡執行你要的操作,當UI更新完成後會自動呼叫這裡面的程式碼                                        }
                                    });

 多謝  記賬本   指出本文的一個BUG,上面應改為:

 

                                    setLoadDataComplete(new isLoadDataListener() {

                                        @Override
                                        
public void loadComplete() {
                                            
// TODO Auto-generated method stub
                                             
//這裡執行你要的操作,當UI更新完成後會自動呼叫這裡面的程式碼                                        }
                                    });

new loadGridAsyncTask(1).execute();

 

       此篇文章希望能對入門不久的Android 開發者有幫助。

相關文章