Android Loader詳解四:回調及完整例子

onLoadFinished
這個方法是在前面已創建的裝載器已經完成其加載過程後被調用.這個方法保證會在應用到裝載器上的數據被釋放之前被調用.在此方法中,你必須刪除所有對舊數據的使用(因為它將很快會被刪除),但是不要自己去釋放它們,因為它們的裝載器會做這些事情.

裝載器一旦瞭解到應用不再使用數據時,將馬上釋放這些數據.例如,如果數據是一個從CursorLoader來的遊標,你不應調用遊標的close().如果遊標被放置在一個CursorAdapter中,你應使用swapCursor()方法,以使舊的遊標不被關閉.例如:

[java] //這個Adapter被用於顯示列表的數據.  
SimpleCursorAdapter mAdapter; 
… 
 
public void onLoadFinished(Loader<Cursor> loader, Cursor data) { 
    // Swap the new cursor in.  (The framework will take care of closing the  
    // old cursor once we return.)  
    mAdapter.swapCursor(data); 

//這個Adapter被用於顯示列表的數據.
SimpleCursorAdapter mAdapter;

public void onLoadFinished(Loader<Cursor> loader, Cursor data) {
    // Swap the new cursor in.  (The framework will take care of closing the
    // old cursor once we return.)
    mAdapter.swapCursor(data);
}

onLoaderReset

當一個已創建的裝載器被重置從而使其數據無效時,此方法被調用.此回調使你能發現什麼時候數據將被釋放於是你可以釋放對它的引用.

下面這個實現調用參數為null的swapCursor():

[java] // 這個Adapter被用於顯示列表的數據.  
SimpleCursorAdapter mAdapter; 
… 
 
public void onLoaderReset(Loader<Cursor> loader) { 
    //此處是用於上面的onLoadFinished()的遊標將被關閉時執行, 我們需確保我們不再使用它.  
    mAdapter.swapCursor(null); 

// 這個Adapter被用於顯示列表的數據.
SimpleCursorAdapter mAdapter;

public void onLoaderReset(Loader<Cursor> loader) {
    //此處是用於上面的onLoadFinished()的遊標將被關閉時執行, 我們需確保我們不再使用它.
    mAdapter.swapCursor(null);
}

例子

作為一個例子,這裡完整實現瞭一個Fragment顯示一個包含從聯系人contentprovider 返回查詢數據的ListView的內容的功能.它使用一個CursorLoader來管理對provider的查詢.

為瞭能從用戶的聯系人中取得數據,本例的manifest必須包含READ_CONTACTS權限.

[java] public static class CursorLoaderListFragment extends ListFragment 
        implements OnQueryTextListener, LoaderManager.LoaderCallbacks<Cursor> { 
 
    // 這是用於顯示列表數據的Adapter  
    SimpleCursorAdapter mAdapter; 
 
    // 如果非null,這是當前的搜索過慮器  
    String mCurFilter; 
 
    @Override public void onActivityCreated(Bundle savedInstanceState) { 
        super.onActivityCreated(savedInstanceState); 
 
        // 如果列表中沒有數據,就給控件一些文字去顯示.在一個真正的應用  
        // 中這應用資源中取得.  
        setEmptyText("No phone numbers"); 
 
        // 我們在動作欄中有一個菜單項.  
        setHasOptionsMenu(true); 
 
        // 創建一個空的adapter,我們將用它顯示加載後的數據  
        mAdapter = new SimpleCursorAdapter(getActivity(), 
                android.R.layout.simple_list_item_2, null, 
                new String[] { Contacts.DISPLAY_NAME, Contacts.CONTACT_STATUS }, 
                new int[] { android.R.id.text1, android.R.id.text2 }, 0); 
        setListAdapter(mAdapter); 
 
        // 準備loader.可能是重連到一個已存在的或開始一個新的  
        getLoaderManager().initLoader(0, null, this); 
    } 
 
    @Override public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) { 
        // 放置一個動作欄項用於搜索.  
        MenuItem item = menu.add("Search"); 
        item.setIcon(android.R.drawable.ic_menu_search); 
        item.setShowAsAction(MenuItem.SHOW_AS_ACTION_IF_ROOM); 
        SearchView sv = new SearchView(getActivity()); 
        sv.setOnQueryTextListener(this); 
        item.setActionView(sv); 
    } 
 
    public boolean onQueryTextChange(String newText) { 
        // 在動作欄上的搜索字串改變時被調用.更新  
        //搜索過濾器,並重啟loader來執行一個新的查詢  
        mCurFilter = !TextUtils.isEmpty(newText) ? newText : null; 
        getLoaderManager().restartLoader(0, null, this); 
        return true; 
    } 
 
    @Override public boolean onQueryTextSubmit(String query) { 
        // 我們不關心這個方法  
        return true; 
    } 
 
    @Override public void onListItemClick(ListView l, View v, int position, long id) { 
        // 寫入你想寫的代碼  
        Log.i("FragmentComplexList", "Item clicked: " + id); 
    } 
 
    // 這是我們想獲取的聯系人中一行的數據.  
    static final String[] CONTACTS_SUMMARY_PROJECTION = new String[] { 
        Contacts._ID, 
        Contacts.DISPLAY_NAME, 
        Contacts.CONTACT_STATUS, 
        Contacts.CONTACT_PRESENCE, 
        Contacts.PHOTO_ID, 
        Contacts.LOOKUP_KEY, 
    }; 
    public Loader<Cursor> onCreateLoader(int id, Bundle args) { 
        // 當一個新的loader需被創建時調用.本例僅有一個Loader,  
        //所以我們不需關心ID.首先設置base URI,URI指向的是聯系人  
        Uri baseUri; 
        if (mCurFilter != null) { 
            baseUri = Uri.withAppendedPath(Contacts.CONTENT_FILTER_URI, 
                    Uri.encode(mCurFilter)); 
        } else { 
            baseUri = Contacts.CONTENT_URI; 
        } 
 
        // 現在創建並返回一個CursorLoader,它將負責創建一個  
        // Cursor用於顯示數據  www.aiwalls.com 
        String select = "((" + Contacts.DISPLAY_NAME + " NOTNULL) AND (" 
                + Contacts.HAS_PHONE_NUMBER + "=1) AND (" 
                + Contacts.DISPLAY_NAME + " != '' ))"; 
        return new CursorLoader(getActivity(), baseUri, 
                CONTACTS_SUMMARY_PROJECTION, select, null, 
                Contacts.DISPLAY_NAME + " COLLATE LOCALIZED ASC"); 
    } 
 
    public void onLoadFinished(Loader<Cursor> loader, Cursor data) { 
        // 將新的cursor換進來.(框架將在我們返回時關心一下舊cursor的關閉)  
        mAdapter.swapCursor(data); 
    } 
 
    public void onLoaderReset(Loader<Cursor> loader) { 
        //在最後一個Cursor準備進入上面的onLoadFinished()之前.  
        // Cursor要被關閉瞭,我們需要確保不再使用它.  
        mAdapter.swapCursor(null); 
    } 

 摘自 nkmnkm的專欄

發佈留言