AsyncTaskLoader не запускается - PullRequest
30 голосов
/ 22 декабря 2011

Я хочу реализовать AsyncTaskLoader в своем проекте с помощью пакета совместимости, поэтому я следовал руководству по загрузчику в Android Docs.

Проблема в том, что загрузчик ничего не делает, кажется, loadInBackground() никогда не вызывается

Есть идеи, что не так в моем коде? (ExpandableListFragment расширяет Fragment, но не переопределяет ни один критический метод)

Спасибо: -)

/ ** EDIT:

Я понял (поздно, я идиот), что AsyncTaskLoader является абстрактным классом, поэтому мне нужно создать его подкласс ... m (__) m Я оставляю вопрос на случай, если кто-то придет за мной, кто знает ...

public class AgendaListFragment extends ExpandableListFragment implements
        LoaderManager.LoaderCallbacks<JSONArray> {

    private TreeMap<Integer, ArrayList<Evento>> mItems = new TreeMap<Integer, ArrayList<Evento>>();
    private AgendaListAdapter mAdapter;
    private ProgressBar mProgressBar;

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
            Bundle savedInstanceState) {
        View root = inflater.inflate(R.layout.fragment_agenda, container);
        mProgressBar = (ProgressBar) root.findViewById(R.id.loading);
        return root;

    }

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

        mAdapter = new AgendaListAdapter(getActivity());
        setListAdapter(mAdapter);

        getLoaderManager().initLoader(0, null, this);

    }

    @Override
    public Loader<JSONArray> onCreateLoader(int arg0, Bundle arg1) {
        mProgressBar.setVisibility(View.VISIBLE);
        return new AsyncTaskLoader<JSONArray>(getActivity()) {
            @Override
            public JSONArray loadInBackground() {

                return getDataFromService(AgendaServices.LISTADO_MES);

            }

        };
    }

    @Override
    public void onLoadFinished(Loader<JSONArray> loader, JSONArray data) {

        // Some stuff to turn JSONArray into TreeMap

        mProgressBar.setVisibility(View.GONE);
        mAdapter.setItems(mItems);

    }

    @Override
    public void onLoaderReset(Loader<JSONArray> arg0) {
        mAdapter.setItems(null);
        mProgressBar.setVisibility(View.VISIBLE);

    }

}

Ответы [ 7 ]

63 голосов
/ 12 января 2012

Я думаю, что лучшее решение для пакета Compatibility - переопределить метод AsyncTaskLoader.onStartLoading .

, например

@Override
protected void onStartLoading() {
  if(dataIsReady) {
    deliverResult(data);
  } else {
    forceLoad();
  }
}
8 голосов
/ 27 декабря 2011

Это точно исправление, но оно должно работать. Я уверен, что библиотека совместимости не работает. Попробуйте это:

getLoaderManager().initLoader(0, null, this).forceLoad();
3 голосов
/ 20 октября 2013

Чок Ян Ченг абсолютно прав:

Проверка на takeContentChanged также кажется важным шагом.

Если вы напишите свой метод так:

protected void onStartLoading() {
    forceLoad();
}

вы заметите, что когда появляется дочерняя активность, а затем вы возвращаетесь к родительской, onStartLoading (и так loadInBackground) вызываются снова!

Что вы можете сделать? Установите для внутренней переменной (mContentChanged) значение true внутри конструктора; затем проверьте эту переменную внутри onStartLoading. Только когда это правда, начните загружать по-настоящему:

package example.util;

import android.content.Context;
import android.support.v4.content.AsyncTaskLoader;

public abstract class ATLoader<D> extends AsyncTaskLoader<D> {

    public ATLoader(Context context) {
        super(context);
        // run only once
        onContentChanged();
    }

    @Override
    protected void onStartLoading() {
        // That's how we start every AsyncTaskLoader...
        // -  code snippet from  android.content.CursorLoader  (method  onStartLoading)
        if (takeContentChanged()) {
            forceLoad();
        }
    }
}
1 голос
/ 19 марта 2013

Глядя на обсуждение на https://code.google.com/p/android/issues/detail?id=14944, проверка на takeContentChanged также кажется важным шагом.

protected void onStartLoading() {
    if (mCursor != null) {
        deliverResult(mCursor);
    }
    if (takeContentChanged() || mCursor == null) {
        forceLoad();
    }
}
0 голосов
/ 17 августа 2016

У меня возникла такая же проблема после перехода с CursorLoader на AsyncTaskLoader.

документация гласит : Подклассы Loader<D> обычно должны реализовывать как минимум onStartLoading (), onStopLoading (), onForceLoad () и onReset () .

AsyncTaskLoader расширяет Loader, но не реализует onStartLoading (), onStopLoading (), onReset () . Вы должны реализовать это самостоятельно!

@ davidshen84 предложил хорошее решение. Я только добавил проверку на takeContentChanged.

@Override
protected void onStartLoading() {
    try {
        if (data != null) {
            deliverResult(data);
        }
        if (takeContentChanged() || data == null) {
            forceLoad();
        }

        Log.d(TAG, "onStartLoading() ");
    } catch (Exception e) {
        Log.d(TAG, e.getMessage());
    }
}

Использование forceLoad () нормально (неплохая практика). Посмотрите, что документация говорит :
Обычно это следует вызывать только при запуске загрузчика, то есть isStarted () возвращает true.

0 голосов
/ 31 января 2014

Я взял исходный код CursorLoader из фреймворка Android и написал класс CustomTaskLoader<T> для облегчения работы.

https://github.com/Palatis/danbooru-gallery-android/blob/new_api/DanbooruGallery/src/main/java/tw/idv/palatis/danboorugallery/android/content/CustomTaskLoader.java

вы в основном реализуете эти две функции:

public abstract T runTaskInBackground(CancellationSignal signal);
public abstract void cleanUp(T oldResult);

посмотрите использование в действиях и фрагментах, например, это: (ну мой код просто игнорирует CancellationSignal, это TODO в моем списке, но вы можете использовать его.)

https://github.com/Palatis/danbooru-gallery-android/blob/new_api/DanbooruGallery/src/main/java/tw/idv/palatis/danboorugallery/PostListFragment.java

return new CustomTaskLoader<Cursor>(getActivity().getApplicationContext()) 
{
    @Override
    public Cursor runTaskInBackground(CancellationSignal signal)
    {
        return SiteSession.getAllPostsCursor(PostListAdapter.POST_COLUMNS);
    }

    @Override
    public void cleanUp(Cursor oldCursor)
    {
        if (!oldCursor.isClosed())
            oldCursor.close();
    }
}
0 голосов
/ 01 июля 2013

У меня все еще была проблема с загрузкой данных. Наконец, я удалил AsyncTaskLoader (версия библиотеки поддержки) и использовал только AsyncTask (не из библиотеки поддержки) для выполнения работы. И это сработало.

Этого тоже может хватить.

Описание и пример: http://developer.android.com/reference/android/os/AsyncTask.html.

Вы должны расширить класс AsyncTask .

Метод doInBackground выполнит работу, а в методе onPostExecute вы получите результат. Для запуска AsyncTask вы вызовете метод execute в его экземпляре. Смотрите ссылку.

...