Android 3.0 - в чем преимущества использования экземпляров LoaderManager? - PullRequest
31 голосов
/ 09 апреля 2011

С 3.0 у нас появился модный LoaderManager, который обрабатывает загрузку данных с использованием AsyncTaskLoader, CursorLoader и других пользовательских Loader экземпляров.Но, читая эти документы, я просто не мог понять, как это лучше, чем просто использовать старый добрый AsyncTask для загрузки данных?

1 Ответ

53 голосов
/ 09 апреля 2011

Ну, их намного проще реализовать, и они позаботятся обо всем, что касается управления жизненным циклом, поэтому они намного менее подвержены ошибкам.

Просто посмотрите на пример кода, чтобы показать результат запроса курсора, который позволяетпользователь интерактивно фильтрует набор результатов через поле ввода запроса на панели действий:

public static class CursorLoaderListFragment extends ListFragment
        implements OnQueryTextListener, LoaderManager.LoaderCallbacks<Cursor> {

    // This is the Adapter being used to display the list's data.
    SimpleCursorAdapter mAdapter;

    // If non-null, this is the current filter the user has provided.
    String mCurFilter;

    @Override public void onActivityCreated(Bundle savedInstanceState) {
        super.onActivityCreated(savedInstanceState);

        // Give some text to display if there is no data.  In a real
        // application this would come from a resource.
        setEmptyText("No phone numbers");

        // We have a menu item to show in action bar.
        setHasOptionsMenu(true);

        // Create an empty adapter we will use to display the loaded data.
        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);

        // Prepare the loader.  Either re-connect with an existing one,
        // or start a new one.
        getLoaderManager().initLoader(0, null, this);
    }

    @Override public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) {
        // Place an action bar item for searching.
        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) {
        // Called when the action bar search text has changed.  Update
        // the search filter, and restart the loader to do a new query
        // with this filter.
        mCurFilter = !TextUtils.isEmpty(newText) ? newText : null;
        getLoaderManager().restartLoader(0, null, this);
        return true;
    }

    @Override public boolean onQueryTextSubmit(String query) {
        // Don't care about this.
        return true;
    }

    @Override public void onListItemClick(ListView l, View v, int position, long id) {
        // Insert desired behavior here.
        Log.i("FragmentComplexList", "Item clicked: " + id);
    }

    // These are the Contacts rows that we will retrieve.
    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) {
        // This is called when a new Loader needs to be created.  This
        // sample only has one Loader, so we don't care about the ID.
        // First, pick the base URI to use depending on whether we are
        // currently filtering.
        Uri baseUri;
        if (mCurFilter != null) {
            baseUri = Uri.withAppendedPath(Contacts.CONTENT_FILTER_URI,
                    Uri.encode(mCurFilter));
        } else {
            baseUri = Contacts.CONTENT_URI;
        }

        // Now create and return a CursorLoader that will take care of
        // creating a Cursor for the data being displayed.
        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) {
        // Swap the new cursor in.  (The framework will take care of closing the
        // old cursor once we return.)
        mAdapter.swapCursor(data);
    }

    public void onLoaderReset(Loader<Cursor> loader) {
        // This is called when the last Cursor provided to onLoadFinished()
        // above is about to be closed.  We need to make sure we are no
        // longer using it.
        mAdapter.swapCursor(null);
    }
}

Правильная реализация этого полного примера самостоятельно с помощью AsyncTask будет включать в себя намного больше кода ... и даже тогдаВы собираетесь реализовать что-то как завершенное и хорошо работающее?Например, будет ли ваша реализация сохранять загруженный Курсор при изменениях конфигурации деятельности, чтобы его не нужно было запрашивать при создании новых экземпляров?LoaderManager / Loader сделает это автоматически за вас, а также позаботится о правильном создании и закрытии курсора на основе жизненного цикла действия.

Также обратите внимание, что для использования этого кода не требуется, чтобы вы вообще думали оубедившись, что длительная работа выполняется из основного потока пользовательского интерфейса.LoaderManager и CursorLoader позаботятся обо всем этом за вас, гарантируя, что вы никогда не заблокируете основной поток при взаимодействии с курсором.Чтобы сделать это правильно, вам действительно нужно иметь два активных объекта Cursor одновременно в точках, чтобы вы могли продолжать отображать интерактивный пользовательский интерфейс с вашим текущим курсором, пока загружается следующий отображаемый объект.LoaderManager делает все это за вас.

Это гораздо более простой API - не нужно знать об AsyncTask и думать о том, что нужно запускать в фоновом режиме, не нужно думать о жизненном цикле активности или о том, какиспользуйте старые API «управляемого курсора» в Activity (которые в любом случае не работали так же хорошо, как LoaderManager).

(Кстати, не забудьте новую статическую библиотеку «support», которая позволяет использовать полный API LoaderManagerна старых версиях Android до 1.6!)

...