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

Using pagination with CursorLoader and MergeCursor closes old cursors

正如标题所说,当尝试对 SimpleCursorAdapterCursorLoader 支持的 ListView 进行分页时,旧游标正在关闭,因此抛出以下异常。前两页加载得很好(第一页没有使用 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)
{
}

这个问题的主要原因是CursorLoader管理了内部游标,所以每当它需要打开一个新的游标(比如一个新的页面查询)时,它就会关闭旧的游标。

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