使用CursorLoader和MergeCursor分页关闭旧游标

时间:2016-01-16 15:24:37

标签: android loader simplecursoradapter android-cursor android-cursorloader

正如标题所说,当试图对由ListViewSimpleCursorAdapter支持的CursorLoader进行分页时,旧游标将被关闭,因此会抛出以下异常。前两页加载得很好(第一页不使用MergeCursor,第二页是第一页使用MergeCursor)。我不会在任何游标上调用任何close()

有趣的是,在调试时,我无法看到任何光标上的关闭标志被设置为true,这是值得的。那么MergeCursor可能会出现问题。如果你们有任何解决方案,请告诉我,我没有想法。

堆栈追踪:

android.database.StaleDataException: Attempting to access a closed CursorWindow.Most probable cause: cursor is deactivated prior to calling this method.
    at android.database.AbstractWindowedCursor.checkPosition(AbstractWindowedCursor.java:139)
    at android.database.AbstractWindowedCursor.getLong(AbstractWindowedCursor.java:74)

代码:

private List<Cursor> mCursorsList = new ArrayList<>();

@Override
public void onScroll(AbsListView view, int firstVisibleItem,
                     int visibleItemCount, int totalItemCount)
{
    if (!isLoading && !isAllLoaded &&
            firstVisibleItem != 0 &&
            firstVisibleItem == totalItemCount - visibleItemCount)
        getActivity().getSupportLoaderManager().restartLoader(LOADER_ID, null, this);
}

@Override
public void onLoadFinished(Loader<Cursor> loader, final Cursor data)
{
        Cursor oldCursor = mCursorAdapter.getCursor();
        mCursorsList.add(data);
        if (oldCursor != null)
        {
            Cursor[] cursorArray = mCursorsList.toArray(new Cursor[mCursorsList.size()]);
            MergeCursor newCursor = new MergeCursor(cursorArray);
            mCursorAdapter.swapCursor(newCursor);
        }
        else // first cursor
        {
            mCursorAdapter.swapCursor(data);
        }
}

@Override
public void onLoaderReset(Loader<Cursor> loader)
{
}

1 个答案:

答案 0 :(得分:1)

此问题的主要原因是CursorLoader管理内部游标,因此无论何时需要打开新游标(例如新页面查询),它都会关闭旧游标。

对于更简单的分页实现,不要使用偏移查询,只是在每个页面上限制,以便新光标也包含所有先前的页面。此外,正如Ian Lake在Google+上建议的那样,有时候您甚至不需要分页,特别是如果您正在进行复杂的连接或对数据进行排序。